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.
Files changed (107) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +84 -0
  3. data/ext/lwtarantool/conn.c +314 -0
  4. data/ext/lwtarantool/depend +23 -0
  5. data/ext/lwtarantool/errors.c +24 -0
  6. data/ext/lwtarantool/extconf.rb +8 -0
  7. data/ext/lwtarantool/lwtarantool.c +12 -0
  8. data/ext/lwtarantool/lwtarantool.h +38 -0
  9. data/ext/lwtarantool/request.c +124 -0
  10. data/ext/lwtarantool/vendor/msgpuck/CMakeLists.txt +73 -0
  11. data/ext/lwtarantool/vendor/msgpuck/hints.c +674 -0
  12. data/ext/lwtarantool/vendor/msgpuck/msgpuck.c +375 -0
  13. data/ext/lwtarantool/vendor/msgpuck/msgpuck.h +2195 -0
  14. data/ext/lwtarantool/vendor/msgpuck/test/CMakeLists.txt +25 -0
  15. data/ext/lwtarantool/vendor/msgpuck/test/msgpuck.c +1126 -0
  16. data/ext/lwtarantool/vendor/msgpuck/test/test.c +105 -0
  17. data/ext/lwtarantool/vendor/msgpuck/test/test.h +123 -0
  18. data/ext/lwtarantool/vendor/tarantool-c/CMakeLists.txt +89 -0
  19. data/ext/lwtarantool/vendor/tarantool-c/cmake/FindMsgPuck.cmake +18 -0
  20. data/ext/lwtarantool/vendor/tarantool-c/cmake/FindSphinx.cmake +10 -0
  21. data/ext/lwtarantool/vendor/tarantool-c/doc/CMakeLists.txt +15 -0
  22. data/ext/lwtarantool/vendor/tarantool-c/doc/source/example.c +340 -0
  23. data/ext/lwtarantool/vendor/tarantool-c/include/CMakeLists.txt +6 -0
  24. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tarantool.h +67 -0
  25. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_auth.h +72 -0
  26. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_buf.h +88 -0
  27. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_call.h +83 -0
  28. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_delete.h +52 -0
  29. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_execute.h +24 -0
  30. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_insert.h +62 -0
  31. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_io.h +67 -0
  32. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iob.h +62 -0
  33. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_iter.h +301 -0
  34. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_mem.h +102 -0
  35. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_net.h +242 -0
  36. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_object.h +248 -0
  37. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_opt.h +138 -0
  38. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_ping.h +49 -0
  39. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_proto.h +295 -0
  40. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_reply.h +204 -0
  41. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_request.h +377 -0
  42. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_schema.h +165 -0
  43. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_select.h +59 -0
  44. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_stream.h +79 -0
  45. data/ext/lwtarantool/vendor/tarantool-c/include/tarantool/tnt_update.h +226 -0
  46. data/ext/lwtarantool/vendor/tarantool-c/include/tp.h +1998 -0
  47. data/ext/lwtarantool/vendor/tarantool-c/test/CMakeLists.txt +36 -0
  48. data/ext/lwtarantool/vendor/tarantool-c/test/common.c +233 -0
  49. data/ext/lwtarantool/vendor/tarantool-c/test/common.h +28 -0
  50. data/ext/lwtarantool/vendor/tarantool-c/test/plain_test.c +152 -0
  51. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_call.c +57 -0
  52. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_disconnect.c +31 -0
  53. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_tcp.c +840 -0
  54. data/ext/lwtarantool/vendor/tarantool-c/test/tarantool_unix.c +384 -0
  55. data/ext/lwtarantool/vendor/tarantool-c/test/test.c +72 -0
  56. data/ext/lwtarantool/vendor/tarantool-c/test/test.h +94 -0
  57. data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.c +6 -0
  58. data/ext/lwtarantool/vendor/tarantool-c/test/tnt_assoc.h +67 -0
  59. data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.c +317 -0
  60. data/ext/lwtarantool/vendor/tarantool-c/third_party/PMurHash.h +68 -0
  61. data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.c +300 -0
  62. data/ext/lwtarantool/vendor/tarantool-c/third_party/base64.h +91 -0
  63. data/ext/lwtarantool/vendor/tarantool-c/third_party/mhash.h +589 -0
  64. data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.c +219 -0
  65. data/ext/lwtarantool/vendor/tarantool-c/third_party/sha1.h +24 -0
  66. data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.c +6528 -0
  67. data/ext/lwtarantool/vendor/tarantool-c/third_party/uri.h +81 -0
  68. data/ext/lwtarantool/vendor/tarantool-c/tnt/CMakeLists.txt +83 -0
  69. data/ext/lwtarantool/vendor/tarantool-c/tnt/pmatomic.h +494 -0
  70. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.c +9 -0
  71. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_assoc.h +172 -0
  72. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_auth.c +118 -0
  73. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_buf.c +171 -0
  74. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_call.c +79 -0
  75. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_delete.c +52 -0
  76. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_execute.c +60 -0
  77. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_insert.c +60 -0
  78. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_io.c +520 -0
  79. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iob.c +75 -0
  80. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_iter.c +293 -0
  81. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_mem.c +82 -0
  82. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_net.c +336 -0
  83. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_object.c +476 -0
  84. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_opt.c +117 -0
  85. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_ping.c +38 -0
  86. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_proto_internal.h +43 -0
  87. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_reply.c +300 -0
  88. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_request.c +336 -0
  89. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_schema.c +334 -0
  90. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_select.c +58 -0
  91. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_stream.c +71 -0
  92. data/ext/lwtarantool/vendor/tarantool-c/tnt/tnt_update.c +247 -0
  93. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/CMakeLists.txt +69 -0
  94. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.c +173 -0
  95. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_dir.h +58 -0
  96. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.c +329 -0
  97. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_log.h +119 -0
  98. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.c +189 -0
  99. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_rpl.h +47 -0
  100. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.c +163 -0
  101. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_snapshot.h +50 -0
  102. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.c +163 -0
  103. data/ext/lwtarantool/vendor/tarantool-c/tntrpl/tnt_xlog.h +50 -0
  104. data/lib/lwtarantool/connection.rb +84 -0
  105. data/lib/lwtarantool/request.rb +58 -0
  106. data/lib/lwtarantool.rb +23 -0
  107. 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
+ }