lwtarantool 0.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/README.md +84 -0
- data/ext/lwtarantool/conn.c +314 -0
- data/ext/lwtarantool/depend +23 -0
- data/ext/lwtarantool/errors.c +24 -0
- data/ext/lwtarantool/extconf.rb +8 -0
- data/ext/lwtarantool/lwtarantool.c +12 -0
- data/ext/lwtarantool/lwtarantool.h +38 -0
- data/ext/lwtarantool/request.c +124 -0
- data/ext/lwtarantool/vendor/msgpuck/CMakeLists.txt +73 -0
- data/ext/lwtarantool/vendor/msgpuck/hints.c +674 -0
- data/ext/lwtarantool/vendor/msgpuck/msgpuck.c +375 -0
- data/ext/lwtarantool/vendor/msgpuck/msgpuck.h +2195 -0
- data/ext/lwtarantool/vendor/msgpuck/test/CMakeLists.txt +25 -0
- data/ext/lwtarantool/vendor/msgpuck/test/msgpuck.c +1126 -0
- data/ext/lwtarantool/vendor/msgpuck/test/test.c +105 -0
- data/ext/lwtarantool/vendor/msgpuck/test/test.h +123 -0
- data/ext/lwtarantool/vendor/tarantool-c/CMakeLists.txt +89 -0
- data/ext/lwtarantool/vendor/tarantool-c/cmake/FindMsgPuck.cmake +18 -0
- data/ext/lwtarantool/vendor/tarantool-c/cmake/FindSphinx.cmake +10 -0
- data/ext/lwtarantool/vendor/tarantool-c/doc/CMakeLists.txt +15 -0
- data/ext/lwtarantool/vendor/tarantool-c/doc/source/example.c +340 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/CMakeLists.txt +6 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tarantool.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_auth.h +72 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_buf.h +88 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_call.h +83 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_delete.h +52 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_execute.h +24 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_insert.h +62 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_io.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iob.h +62 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iter.h +301 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_mem.h +102 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_net.h +242 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_object.h +248 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_opt.h +138 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_ping.h +49 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_proto.h +295 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_reply.h +204 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_request.h +377 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_schema.h +165 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_select.h +59 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_stream.h +79 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_update.h +226 -0
- data/ext/lwtarantool/vendor/tarantool-c/include/tp.h +1998 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/CMakeLists.txt +36 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/common.c +233 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/common.h +28 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/plain_test.c +152 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_call.c +57 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_disconnect.c +31 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_tcp.c +840 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_unix.c +384 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/test.c +72 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/test.h +94 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.c +6 -0
- data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.h +67 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.c +317 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.h +68 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.c +300 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.h +91 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/mhash.h +589 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.c +219 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.h +24 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.c +6528 -0
- data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.h +81 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/CMakeLists.txt +83 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/pmatomic.h +494 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.c +9 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.h +172 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_auth.c +118 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_buf.c +171 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_call.c +79 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_delete.c +52 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_execute.c +60 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_insert.c +60 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_io.c +520 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iob.c +75 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iter.c +293 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_mem.c +82 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_net.c +336 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_object.c +476 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_opt.c +117 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_ping.c +38 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_proto_internal.h +43 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_reply.c +300 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_request.c +336 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_schema.c +334 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_select.c +58 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_stream.c +71 -0
- data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_update.c +247 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/CMakeLists.txt +69 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.c +173 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.h +58 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.c +329 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.h +119 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.c +189 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.h +47 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.c +163 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.h +50 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.c +163 -0
- data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.h +50 -0
- data/lib/lwtarantool/connection.rb +84 -0
- data/lib/lwtarantool/request.rb +58 -0
- data/lib/lwtarantool.rb +23 -0
- metadata +164 -0
@@ -0,0 +1,69 @@
|
|
1
|
+
#============================================================================#
|
2
|
+
# build flags
|
3
|
+
#============================================================================#
|
4
|
+
|
5
|
+
# default flags
|
6
|
+
if (${CMAKE_BUILD_TYPE} STREQUAL "None")
|
7
|
+
set (tntrpl_cflags "-std=gnu99")
|
8
|
+
else()
|
9
|
+
set (tntrpl_cflags "-std=gnu99 -Wall -Wextra")
|
10
|
+
set (tntrpl_cflags "${tntrpl_cflags} -Wno-sign-compare -Wno-strict-aliasing")
|
11
|
+
endif()
|
12
|
+
|
13
|
+
# Only add -Werror if it's a debug build, done by developers.
|
14
|
+
if (${CMAKE_BUILD_TYPE} STREQUAL "Debug")
|
15
|
+
set (tntrpl_cflags "${tntrpl_cflags} -Werror")
|
16
|
+
endif()
|
17
|
+
|
18
|
+
#============================================================================#
|
19
|
+
# Build tnt rpl project
|
20
|
+
#============================================================================#
|
21
|
+
|
22
|
+
#
|
23
|
+
# source files
|
24
|
+
#
|
25
|
+
|
26
|
+
set (tntrpl_sources tnt_log.c tnt_dir.c tnt_xlog.c tnt_snapshot.c tnt_rpl.c
|
27
|
+
${CMAKE_SOURCE_DIR}/third_party/crc32.c)
|
28
|
+
|
29
|
+
#----------------------------------------------------------------------------#
|
30
|
+
# Builds
|
31
|
+
#----------------------------------------------------------------------------#
|
32
|
+
|
33
|
+
# Here we manage to build static/dynamic libraries ourselves,
|
34
|
+
# do not use the top level settings.
|
35
|
+
string(REPLACE "-static-libgcc" "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}")
|
36
|
+
string(REPLACE "-static" "" CMAKE_C_FLAGS "${CMAKE_C_FLAGS}")
|
37
|
+
|
38
|
+
if (CMAKE_COMPILER_IS_GNUCC AND NOT CMAKE_COMPILER_IS_CLANG)
|
39
|
+
set (tnt_cflags "${tnt_cflags} -static-libgcc")
|
40
|
+
endif()
|
41
|
+
|
42
|
+
#
|
43
|
+
# Static library
|
44
|
+
#
|
45
|
+
|
46
|
+
project(tntrpl)
|
47
|
+
add_library(tntrpl STATIC ${tntrpl_sources})
|
48
|
+
set_target_properties(tntrpl PROPERTIES COMPILE_FLAGS "${tntrpl_cflags}")
|
49
|
+
set_target_properties(tntrpl PROPERTIES VERSION ${LIBTNT_VERSION} SOVERSION ${LIBTNT_SOVERSION})
|
50
|
+
set_target_properties(tntrpl PROPERTIES OUTPUT_NAME "tarantoolrpl")
|
51
|
+
|
52
|
+
#
|
53
|
+
# Shared library
|
54
|
+
#
|
55
|
+
|
56
|
+
project(tntrpl_shared)
|
57
|
+
add_library(tntrpl_shared SHARED ${tntrpl_sources})
|
58
|
+
target_link_libraries(tntrpl_shared tnt_shared tntnet_shared)
|
59
|
+
set_target_properties(tntrpl_shared PROPERTIES OUTPUT_NAME tntrpl)
|
60
|
+
set_target_properties(tntrpl_shared PROPERTIES COMPILE_FLAGS "${tntrpl_cflags}")
|
61
|
+
set_target_properties(tntrpl_shared PROPERTIES VERSION ${LIBTNT_VERSION} SOVERSION ${LIBTNT_SOVERSION})
|
62
|
+
set_target_properties(tntrpl_shared PROPERTIES OUTPUT_NAME "tarantoolrpl")
|
63
|
+
|
64
|
+
#----------------------------------------------------------------------------#
|
65
|
+
# Install
|
66
|
+
#----------------------------------------------------------------------------#
|
67
|
+
|
68
|
+
install (TARGETS tntrpl ARCHIVE DESTINATION ${CMAKE_INSTALL_LIBDIR})
|
69
|
+
install (TARGETS tntrpl_shared LIBRARY DESTINATION ${CMAKE_INSTALL_LIBDIR})
|
@@ -0,0 +1,173 @@
|
|
1
|
+
|
2
|
+
/*
|
3
|
+
* Redistribution and use in source and binary forms, with or
|
4
|
+
* without modification, are permitted provided that the following
|
5
|
+
* conditions are met:
|
6
|
+
*
|
7
|
+
* 1. Redistributions of source code must retain the above
|
8
|
+
* copyright notice, this list of conditions and the
|
9
|
+
* following disclaimer.
|
10
|
+
*
|
11
|
+
* 2. Redistributions in binary form must reproduce the above
|
12
|
+
* copyright notice, this list of conditions and the following
|
13
|
+
* disclaimer in the documentation and/or other materials
|
14
|
+
* provided with the distribution.
|
15
|
+
*
|
16
|
+
* THIS SOFTWARE IS PROVIDED BY <COPYRIGHT HOLDER> ``AS IS'' AND
|
17
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
18
|
+
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
19
|
+
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
20
|
+
* <COPYRIGHT HOLDER> OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
21
|
+
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
22
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
23
|
+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
24
|
+
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
25
|
+
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
26
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
27
|
+
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
28
|
+
* SUCH DAMAGE.
|
29
|
+
*/
|
30
|
+
|
31
|
+
#include <stdlib.h>
|
32
|
+
#include <stdio.h>
|
33
|
+
#include <stdint.h>
|
34
|
+
#include <string.h>
|
35
|
+
#include <limits.h>
|
36
|
+
|
37
|
+
#include <unistd.h>
|
38
|
+
#include <sys/types.h>
|
39
|
+
#include <dirent.h>
|
40
|
+
#include <errno.h>
|
41
|
+
|
42
|
+
#include <tarantool.h>
|
43
|
+
#include <tnt_dir.h>
|
44
|
+
|
45
|
+
void tnt_dir_init(struct tnt_dir *d, enum tnt_dir_type type) {
|
46
|
+
d->type = type;
|
47
|
+
d->path = NULL;
|
48
|
+
d->files = NULL;
|
49
|
+
d->count = 0;
|
50
|
+
}
|
51
|
+
|
52
|
+
void tnt_dir_free(struct tnt_dir *d) {
|
53
|
+
if (d->path) {
|
54
|
+
tnt_mem_free(d->path);
|
55
|
+
d->path = NULL;
|
56
|
+
}
|
57
|
+
if (d->files) {
|
58
|
+
int i = 0;
|
59
|
+
while (i < d->count) {
|
60
|
+
if (d->files[i].name)
|
61
|
+
tnt_mem_free(d->files[i].name);
|
62
|
+
i++;
|
63
|
+
}
|
64
|
+
tnt_mem_free(d->files);
|
65
|
+
d->files = NULL;
|
66
|
+
}
|
67
|
+
}
|
68
|
+
|
69
|
+
static int tnt_dir_put(struct tnt_dir *d,
|
70
|
+
int *top, char *name, uint64_t lsn)
|
71
|
+
{
|
72
|
+
if (d->count == *top) {
|
73
|
+
*top = (*top == 0) ? 128 : *top * 2;
|
74
|
+
d->files = tnt_mem_realloc(d->files, sizeof(struct tnt_dir_file) * *top);
|
75
|
+
if (d->files == NULL)
|
76
|
+
return -1;
|
77
|
+
}
|
78
|
+
struct tnt_dir_file *current = &d->files[d->count];
|
79
|
+
current->lsn = lsn;
|
80
|
+
current->name = (char*)tnt_mem_dup(name);
|
81
|
+
if (current->name == NULL)
|
82
|
+
return -1;
|
83
|
+
d->count++;
|
84
|
+
return 0;
|
85
|
+
}
|
86
|
+
|
87
|
+
static int tnt_dir_cmp(const void *_a, const void *_b) {
|
88
|
+
const struct tnt_dir_file *a = _a;
|
89
|
+
const struct tnt_dir_file *b = _b;
|
90
|
+
if (a->lsn == b->lsn)
|
91
|
+
return 0;
|
92
|
+
return (a->lsn > b->lsn) ? 1: -1;
|
93
|
+
}
|
94
|
+
|
95
|
+
int tnt_dir_scan(struct tnt_dir *d, char *path) {
|
96
|
+
d->path = tnt_mem_dup(path);
|
97
|
+
if (d->path == NULL)
|
98
|
+
return -1;
|
99
|
+
DIR *dir = opendir(d->path);
|
100
|
+
if (dir == NULL)
|
101
|
+
goto error;
|
102
|
+
|
103
|
+
struct dirent *dep = NULL;
|
104
|
+
struct dirent de;
|
105
|
+
int rc, top = 0;
|
106
|
+
while ((rc = readdir_r(dir, &de, &dep)) == 0) {
|
107
|
+
if (dep == NULL)
|
108
|
+
break;
|
109
|
+
if (strcmp(de.d_name, ".") == 0 ||
|
110
|
+
strcmp(de.d_name, "..") == 0)
|
111
|
+
continue;
|
112
|
+
|
113
|
+
char *ext = strchr(de.d_name, '.');
|
114
|
+
if (ext == NULL)
|
115
|
+
continue;
|
116
|
+
|
117
|
+
switch (d->type) {
|
118
|
+
case TNT_DIR_XLOG:
|
119
|
+
if (strcmp(ext, ".xlog") != 0)
|
120
|
+
continue;
|
121
|
+
break;
|
122
|
+
case TNT_DIR_SNAPSHOT:
|
123
|
+
if (strcmp(ext, ".snap") != 0)
|
124
|
+
continue;
|
125
|
+
break;
|
126
|
+
}
|
127
|
+
|
128
|
+
uint64_t lsn = strtoll(de.d_name, &ext, 10);
|
129
|
+
if (lsn == LLONG_MAX || lsn == LLONG_MIN)
|
130
|
+
continue;
|
131
|
+
|
132
|
+
rc = tnt_dir_put(d, &top, de.d_name, lsn);
|
133
|
+
if (rc == -1)
|
134
|
+
goto error;
|
135
|
+
}
|
136
|
+
if (rc != 0)
|
137
|
+
goto error;
|
138
|
+
|
139
|
+
qsort(d->files, d->count, sizeof(struct tnt_dir_file),
|
140
|
+
tnt_dir_cmp);
|
141
|
+
|
142
|
+
closedir(dir);
|
143
|
+
return 0;
|
144
|
+
error:
|
145
|
+
if (dir)
|
146
|
+
closedir(dir);
|
147
|
+
tnt_dir_free(d);
|
148
|
+
return -1;
|
149
|
+
}
|
150
|
+
|
151
|
+
int tnt_dir_match_gt(struct tnt_dir *d, uint64_t *out) {
|
152
|
+
if (d->count == 0)
|
153
|
+
return -1;
|
154
|
+
*out = d->files[d->count -1].lsn;
|
155
|
+
return 0;
|
156
|
+
}
|
157
|
+
|
158
|
+
int tnt_dir_match_inc(struct tnt_dir *d, uint64_t lsn, uint64_t *out) {
|
159
|
+
if (d->count == 0)
|
160
|
+
return -1;
|
161
|
+
int current = 0;
|
162
|
+
int count = d->count;
|
163
|
+
while (count > 1) {
|
164
|
+
if (d->files[current].lsn <= lsn && lsn <= d->files[current + 1].lsn) {
|
165
|
+
*out = d->files[current].lsn;
|
166
|
+
return 0;
|
167
|
+
}
|
168
|
+
current++;
|
169
|
+
count--;
|
170
|
+
}
|
171
|
+
*out = d->files[current].lsn;
|
172
|
+
return 0;
|
173
|
+
}
|
@@ -0,0 +1,58 @@
|
|
1
|
+
#ifndef TNT_DIR_H_INCLUDED
|
2
|
+
#define TNT_DIR_H_INCLUDED
|
3
|
+
|
4
|
+
/*
|
5
|
+
* Redistribution and use in source and binary forms, with or
|
6
|
+
* without modification, are permitted provided that the following
|
7
|
+
* conditions are met:
|
8
|
+
*
|
9
|
+
* 1. Redistributions of source code must retain the above
|
10
|
+
* copyright notice, this list of conditions and the
|
11
|
+
* following disclaimer.
|
12
|
+
*
|
13
|
+
* 2. Redistributions in binary form must reproduce the above
|
14
|
+
* copyright notice, this list of conditions and the following
|
15
|
+
* disclaimer in the documentation and/or other materials
|
16
|
+
* provided with the distribution.
|
17
|
+
*
|
18
|
+
* THIS SOFTWARE IS PROVIDED BY <COPYRIGHT HOLDER> ``AS IS'' AND
|
19
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
20
|
+
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
21
|
+
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
22
|
+
* <COPYRIGHT HOLDER> OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
23
|
+
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
24
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
25
|
+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
26
|
+
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
27
|
+
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
28
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
29
|
+
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
30
|
+
* SUCH DAMAGE.
|
31
|
+
*/
|
32
|
+
|
33
|
+
enum tnt_dir_type {
|
34
|
+
TNT_DIR_XLOG,
|
35
|
+
TNT_DIR_SNAPSHOT
|
36
|
+
};
|
37
|
+
|
38
|
+
struct tnt_dir_file {
|
39
|
+
uint64_t lsn;
|
40
|
+
char *name;
|
41
|
+
};
|
42
|
+
|
43
|
+
struct tnt_dir {
|
44
|
+
enum tnt_dir_type type;
|
45
|
+
char *path;
|
46
|
+
struct tnt_dir_file *files;
|
47
|
+
int count;
|
48
|
+
};
|
49
|
+
|
50
|
+
void tnt_dir_init(struct tnt_dir *d, enum tnt_dir_type type);
|
51
|
+
void tnt_dir_free(struct tnt_dir *d);
|
52
|
+
|
53
|
+
int tnt_dir_scan(struct tnt_dir *d, char *path);
|
54
|
+
|
55
|
+
int tnt_dir_match_gt(struct tnt_dir *d, uint64_t *out);
|
56
|
+
int tnt_dir_match_inc(struct tnt_dir *d, uint64_t lsn, uint64_t *out);
|
57
|
+
|
58
|
+
#endif
|
@@ -0,0 +1,329 @@
|
|
1
|
+
|
2
|
+
/*
|
3
|
+
* Redistribution and use in source and binary forms, with or
|
4
|
+
* without modification, are permitted provided that the following
|
5
|
+
* conditions are met:
|
6
|
+
*
|
7
|
+
* 1. Redistributions of source code must retain the above
|
8
|
+
* copyright notice, this list of conditions and the
|
9
|
+
* following disclaimer.
|
10
|
+
*
|
11
|
+
* 2. Redistributions in binary form must reproduce the above
|
12
|
+
* copyright notice, this list of conditions and the following
|
13
|
+
* disclaimer in the documentation and/or other materials
|
14
|
+
* provided with the distribution.
|
15
|
+
*
|
16
|
+
* THIS SOFTWARE IS PROVIDED BY <COPYRIGHT HOLDER> ``AS IS'' AND
|
17
|
+
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
|
18
|
+
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
19
|
+
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL
|
20
|
+
* <COPYRIGHT HOLDER> OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
21
|
+
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
22
|
+
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
|
23
|
+
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR
|
24
|
+
* BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
|
25
|
+
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
26
|
+
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
|
27
|
+
* THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
28
|
+
* SUCH DAMAGE.
|
29
|
+
*/
|
30
|
+
|
31
|
+
#include <stdlib.h>
|
32
|
+
#include <stdio.h>
|
33
|
+
#include <stdarg.h>
|
34
|
+
#include <string.h>
|
35
|
+
|
36
|
+
#include <unistd.h>
|
37
|
+
#include <fcntl.h>
|
38
|
+
#include <errno.h>
|
39
|
+
|
40
|
+
#include <third_party/crc32.h>
|
41
|
+
|
42
|
+
#include <tarantool.h>
|
43
|
+
#include <tnt_log.h>
|
44
|
+
|
45
|
+
enum tnt_log_type tnt_log_guess(char *file) {
|
46
|
+
if (file == NULL)
|
47
|
+
return TNT_LOG_XLOG;
|
48
|
+
char *ext = strrchr(file, '.');
|
49
|
+
if (ext == NULL)
|
50
|
+
return TNT_LOG_NONE;
|
51
|
+
if (strcasecmp(ext, ".snap") == 0)
|
52
|
+
return TNT_LOG_SNAPSHOT;
|
53
|
+
if (strcasecmp(ext, ".xlog") == 0)
|
54
|
+
return TNT_LOG_XLOG;
|
55
|
+
return TNT_LOG_NONE;
|
56
|
+
}
|
57
|
+
|
58
|
+
inline static int
|
59
|
+
tnt_log_seterr(struct tnt_log *l, enum tnt_log_error e) {
|
60
|
+
l->error = e;
|
61
|
+
if (e == TNT_LOG_ESYSTEM)
|
62
|
+
l->errno_ = errno;
|
63
|
+
return -1;
|
64
|
+
}
|
65
|
+
|
66
|
+
const uint32_t tnt_log_marker_v11 = 0xba0babed;
|
67
|
+
const uint32_t tnt_log_marker_eof_v11 = 0x10adab1e;
|
68
|
+
|
69
|
+
inline static int
|
70
|
+
tnt_log_eof(struct tnt_log *l, char *data) {
|
71
|
+
uint32_t marker = 0;
|
72
|
+
if (data)
|
73
|
+
tnt_mem_free(data);
|
74
|
+
/* checking eof condition */
|
75
|
+
if (ftello(l->fd) == l->offset + sizeof(tnt_log_marker_eof_v11)) {
|
76
|
+
fseeko(l->fd, l->offset, SEEK_SET);
|
77
|
+
if (fread(&marker, sizeof(marker), 1, l->fd) != 1)
|
78
|
+
return tnt_log_seterr(l, TNT_LOG_ESYSTEM);
|
79
|
+
else
|
80
|
+
if (marker != tnt_log_marker_eof_v11)
|
81
|
+
return tnt_log_seterr(l, TNT_LOG_ECORRUPT);
|
82
|
+
l->offset = ftello(l->fd);
|
83
|
+
}
|
84
|
+
/* eof */
|
85
|
+
return 1;
|
86
|
+
}
|
87
|
+
|
88
|
+
static int tnt_log_read(struct tnt_log *l, char **buf, uint32_t *size)
|
89
|
+
{
|
90
|
+
/* current record offset (before marker) */
|
91
|
+
l->current_offset = ftello(l->fd);
|
92
|
+
|
93
|
+
/* reading marker */
|
94
|
+
char *data = NULL;
|
95
|
+
uint32_t marker = 0;
|
96
|
+
if (fread(&marker, sizeof(marker), 1, l->fd) != 1)
|
97
|
+
return tnt_log_eof(l, data);
|
98
|
+
|
99
|
+
/* seeking for marker if necessary */
|
100
|
+
while (marker != tnt_log_marker_v11) {
|
101
|
+
int c = fgetc(l->fd);
|
102
|
+
if (c == EOF)
|
103
|
+
return tnt_log_eof(l, data);
|
104
|
+
marker = marker >> 8 | ((uint32_t) c & 0xff) <<
|
105
|
+
(sizeof(marker) * 8 - 8);
|
106
|
+
}
|
107
|
+
|
108
|
+
/* reading header */
|
109
|
+
if (fread(&l->current.hdr, sizeof(l->current.hdr), 1, l->fd) != 1)
|
110
|
+
return tnt_log_eof(l, data);
|
111
|
+
|
112
|
+
/* updating offset */
|
113
|
+
l->offset = ftello(l->fd);
|
114
|
+
|
115
|
+
/* checking header crc, starting from lsn */
|
116
|
+
uint32_t crc32_hdr =
|
117
|
+
crc32c(0, (unsigned char*)&l->current.hdr + sizeof(uint32_t),
|
118
|
+
sizeof(struct tnt_log_header_v11) -
|
119
|
+
sizeof(uint32_t));
|
120
|
+
if (crc32_hdr != l->current.hdr.crc32_hdr)
|
121
|
+
return tnt_log_seterr(l, TNT_LOG_ECORRUPT);
|
122
|
+
|
123
|
+
/* allocating memory and reading data */
|
124
|
+
data = tnt_mem_alloc(l->current.hdr.len);
|
125
|
+
if (data == NULL)
|
126
|
+
return tnt_log_seterr(l, TNT_LOG_EMEMORY);
|
127
|
+
if (fread(data, l->current.hdr.len, 1, l->fd) != 1)
|
128
|
+
return tnt_log_eof(l, data);
|
129
|
+
|
130
|
+
/* checking data crc */
|
131
|
+
uint32_t crc32_data = crc32c(0, (unsigned char*)data, l->current.hdr.len);
|
132
|
+
if (crc32_data != l->current.hdr.crc32_data) {
|
133
|
+
tnt_mem_free(data);
|
134
|
+
return tnt_log_seterr(l, TNT_LOG_ECORRUPT);
|
135
|
+
}
|
136
|
+
|
137
|
+
*buf = data;
|
138
|
+
*size = l->current.hdr.len;
|
139
|
+
return 0;
|
140
|
+
}
|
141
|
+
|
142
|
+
static int
|
143
|
+
tnt_log_process_xlog(struct tnt_log *l, char *buf, uint32_t size,
|
144
|
+
union tnt_log_value *value)
|
145
|
+
{
|
146
|
+
(void)size;
|
147
|
+
/* copying row data */
|
148
|
+
memcpy(&l->current.row, buf, sizeof(l->current.row));
|
149
|
+
|
150
|
+
/* preparing pseudo iproto header */
|
151
|
+
struct tnt_header hdr_iproto;
|
152
|
+
hdr_iproto.type = l->current.row.op;
|
153
|
+
hdr_iproto.len = l->current.hdr.len - sizeof(l->current.row);
|
154
|
+
hdr_iproto.reqid = 0;
|
155
|
+
|
156
|
+
/* deserializing operation */
|
157
|
+
tnt_request_init(&value->r);
|
158
|
+
size_t off = 0;
|
159
|
+
int rc = tnt_request(&value->r,
|
160
|
+
buf + sizeof(l->current.row),
|
161
|
+
l->current.hdr.len - sizeof(l->current.row),
|
162
|
+
&off,
|
163
|
+
&hdr_iproto);
|
164
|
+
|
165
|
+
/* in case of not completed request or parsing error */
|
166
|
+
if (rc != 0)
|
167
|
+
return tnt_log_seterr(l, TNT_LOG_ECORRUPT);
|
168
|
+
return 0;
|
169
|
+
}
|
170
|
+
|
171
|
+
static int
|
172
|
+
tnt_log_process_snapshot(struct tnt_log *l, char *buf, uint32_t size,
|
173
|
+
union tnt_log_value *value)
|
174
|
+
{
|
175
|
+
(void)size;
|
176
|
+
|
177
|
+
/* freeing previously allocated tuple */
|
178
|
+
tnt_tuple_free(&value->t);
|
179
|
+
|
180
|
+
/* copying snapshot row data */
|
181
|
+
memcpy(&l->current.row_snap, buf, sizeof(l->current.row_snap));
|
182
|
+
|
183
|
+
/* reading and validating tuple */
|
184
|
+
struct tnt_tuple *tu =
|
185
|
+
tnt_tuple_set_as(&value->t, buf + sizeof(l->current.row_snap),
|
186
|
+
l->current.row_snap.data_size,
|
187
|
+
l->current.row_snap.tuple_size);
|
188
|
+
if (tu == NULL)
|
189
|
+
return tnt_log_seterr(l, TNT_LOG_ECORRUPT);
|
190
|
+
|
191
|
+
return (tu) ? 0 : -1;
|
192
|
+
}
|
193
|
+
|
194
|
+
struct tnt_log_row*
|
195
|
+
tnt_log_next_to(struct tnt_log *l, union tnt_log_value *value) {
|
196
|
+
char *buf = NULL;
|
197
|
+
uint32_t size = 0;
|
198
|
+
int rc = l->read(l, &buf, &size);
|
199
|
+
if (rc != 0)
|
200
|
+
return NULL;
|
201
|
+
rc = l->process(l, buf, size, value);
|
202
|
+
if (rc != 0) {
|
203
|
+
tnt_mem_free(buf);
|
204
|
+
return NULL;
|
205
|
+
}
|
206
|
+
if (l->type == TNT_LOG_XLOG) {
|
207
|
+
tnt_request_setorigin(&value->r, buf, size);
|
208
|
+
} else {
|
209
|
+
tnt_mem_free(buf);
|
210
|
+
}
|
211
|
+
return &l->current;
|
212
|
+
}
|
213
|
+
|
214
|
+
struct tnt_log_row *tnt_log_next(struct tnt_log *l) {
|
215
|
+
return tnt_log_next_to(l, &l->current_value);
|
216
|
+
}
|
217
|
+
|
218
|
+
inline static int
|
219
|
+
tnt_log_open_err(struct tnt_log *l, enum tnt_log_error e) {
|
220
|
+
tnt_log_seterr(l, e);
|
221
|
+
tnt_log_close(l);
|
222
|
+
return -1;
|
223
|
+
}
|
224
|
+
|
225
|
+
enum tnt_log_error
|
226
|
+
tnt_log_open(struct tnt_log *l, char *file, enum tnt_log_type type)
|
227
|
+
{
|
228
|
+
char filetype[32];
|
229
|
+
char version[32];
|
230
|
+
char *rc, *magic = "\0";
|
231
|
+
l->type = type;
|
232
|
+
/* trying to open file */
|
233
|
+
if (file) {
|
234
|
+
l->fd = fopen(file, "r");
|
235
|
+
if (l->fd == NULL)
|
236
|
+
return tnt_log_open_err(l, TNT_LOG_ESYSTEM);
|
237
|
+
} else {
|
238
|
+
l->fd = stdin;
|
239
|
+
}
|
240
|
+
/* reading xlog filetype */
|
241
|
+
rc = fgets(filetype, sizeof(filetype), l->fd);
|
242
|
+
if (rc == NULL)
|
243
|
+
return tnt_log_open_err(l, TNT_LOG_ESYSTEM);
|
244
|
+
/* reading log version */
|
245
|
+
rc = fgets(version, sizeof(version), l->fd);
|
246
|
+
if (rc == NULL)
|
247
|
+
return tnt_log_open_err(l, TNT_LOG_ESYSTEM);
|
248
|
+
/* checking file type and setting read/process
|
249
|
+
* interfaces */
|
250
|
+
l->read = tnt_log_read;
|
251
|
+
switch (type) {
|
252
|
+
case TNT_LOG_XLOG:
|
253
|
+
magic = TNT_LOG_MAGIC_XLOG;
|
254
|
+
l->process = tnt_log_process_xlog;
|
255
|
+
break;
|
256
|
+
case TNT_LOG_SNAPSHOT:
|
257
|
+
magic = TNT_LOG_MAGIC_SNAP;
|
258
|
+
l->process = tnt_log_process_snapshot;
|
259
|
+
break;
|
260
|
+
case TNT_LOG_NONE:
|
261
|
+
break;
|
262
|
+
}
|
263
|
+
if (strcmp(filetype, magic))
|
264
|
+
return tnt_log_open_err(l, TNT_LOG_ETYPE);
|
265
|
+
/* checking version */
|
266
|
+
if (strcmp(version, TNT_LOG_VERSION))
|
267
|
+
return tnt_log_open_err(l, TNT_LOG_EVERSION);
|
268
|
+
for (;;) {
|
269
|
+
char buf[256];
|
270
|
+
rc = fgets(buf, sizeof(buf), l->fd);
|
271
|
+
if (rc == NULL)
|
272
|
+
return tnt_log_open_err(l, TNT_LOG_EFAIL);
|
273
|
+
if (strcmp(rc, "\n") == 0 || strcmp(rc, "\r\n") == 0)
|
274
|
+
break;
|
275
|
+
}
|
276
|
+
/* getting current offset */
|
277
|
+
l->offset = ftello(l->fd);
|
278
|
+
l->current_offset = 0;
|
279
|
+
memset(&l->current_value, 0, sizeof(l->current_value));
|
280
|
+
return 0;
|
281
|
+
}
|
282
|
+
|
283
|
+
void tnt_log_close(struct tnt_log *l) {
|
284
|
+
if (l->fd && l->fd != stdin)
|
285
|
+
fclose(l->fd);
|
286
|
+
l->fd = NULL;
|
287
|
+
}
|
288
|
+
|
289
|
+
int tnt_log_seek(struct tnt_log *l, off_t offset)
|
290
|
+
{
|
291
|
+
l->offset = offset;
|
292
|
+
return fseeko(l->fd, offset, SEEK_SET);
|
293
|
+
}
|
294
|
+
|
295
|
+
enum tnt_log_error tnt_log_error(struct tnt_log *l) {
|
296
|
+
return l->error;
|
297
|
+
}
|
298
|
+
|
299
|
+
struct tnt_log_error_desc {
|
300
|
+
enum tnt_log_error type;
|
301
|
+
char *desc;
|
302
|
+
};
|
303
|
+
|
304
|
+
static struct tnt_log_error_desc tnt_log_error_list[] =
|
305
|
+
{
|
306
|
+
{ TNT_LOG_EOK, "ok" },
|
307
|
+
{ TNT_LOG_EFAIL, "fail" },
|
308
|
+
{ TNT_LOG_EMEMORY, "memory allocation failed" },
|
309
|
+
{ TNT_LOG_ETYPE, "file type mismatch" },
|
310
|
+
{ TNT_LOG_EVERSION, "file version mismatch" },
|
311
|
+
{ TNT_LOG_ECORRUPT, "file crc failed or bad eof marker" },
|
312
|
+
{ TNT_LOG_ESYSTEM, "system error" },
|
313
|
+
{ TNT_LOG_LAST, NULL }
|
314
|
+
};
|
315
|
+
|
316
|
+
char *tnt_log_strerror(struct tnt_log *l) {
|
317
|
+
if (l->error == TNT_LOG_ESYSTEM) {
|
318
|
+
static char msg[256];
|
319
|
+
snprintf(msg, sizeof(msg), "%s (errno: %d)",
|
320
|
+
strerror(l->errno_),
|
321
|
+
l->errno_);
|
322
|
+
return msg;
|
323
|
+
}
|
324
|
+
return tnt_log_error_list[(int)l->error].desc;
|
325
|
+
}
|
326
|
+
|
327
|
+
int tnt_log_errno(struct tnt_log *l) {
|
328
|
+
return l->errno_;
|
329
|
+
}
|