mdbx 0.1.0.pre.20201217111933
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
- checksums.yaml.gz.sig +0 -0
- data.tar.gz.sig +0 -0
- data/History.md +9 -0
- data/README.md +129 -0
- data/ext/mdbx_ext/database.c +511 -0
- data/ext/mdbx_ext/extconf.rb +11 -0
- data/ext/mdbx_ext/mdbx_ext.c +26 -0
- data/ext/mdbx_ext/mdbx_ext.h +29 -0
- data/lib/mdbx.rb +20 -0
- data/lib/mdbx/database.rb +68 -0
- data/lib/mdbx_ext.so +0 -0
- metadata +208 -0
- metadata.gz.sig +2 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: be5af4b8718ccda99471808092ab76ed497758add8a8c429d6eae16b9279c007
|
4
|
+
data.tar.gz: a183a3ba5019cc1b8cbda9ae45f803342c02011de762202728b6ab96c8f7d133
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: e7e463a36ed38653930e9894404431de494cc39e6efa2ae6993681d4198b28c9ecf0c160b09f01353d6296699e9a2ea780973f983a7a32b91b6630d76ad10e2a
|
7
|
+
data.tar.gz: 300534e3d3c342019d803840e956afdc7d4e3bb114d8fe4df95918cfb85808df146979ff56af539e23ae8bfbacf9f7acab1aaaef4742b0b16a162ad6364d7bba
|
checksums.yaml.gz.sig
ADDED
Binary file
|
data.tar.gz.sig
ADDED
Binary file
|
data/History.md
ADDED
data/README.md
ADDED
@@ -0,0 +1,129 @@
|
|
1
|
+
# Ruby MDBX
|
2
|
+
|
3
|
+
home
|
4
|
+
: https://code.martini.nu/ruby-mdbx
|
5
|
+
|
6
|
+
code
|
7
|
+
: https://code.martini.nu/ruby-mdbx
|
8
|
+
|
9
|
+
docs
|
10
|
+
: https://martini.nu/docs/ruby-mdbx
|
11
|
+
|
12
|
+
github
|
13
|
+
: https://github.com/mahlon/ruby-mdbx
|
14
|
+
|
15
|
+
gitlab
|
16
|
+
: https://gitlab.com/mahlon/ruby-mdbx
|
17
|
+
|
18
|
+
sourcehut:
|
19
|
+
: https://hg.sr.ht/~mahlon/ruby-mdbx
|
20
|
+
|
21
|
+
|
22
|
+
## Description
|
23
|
+
|
24
|
+
This is a Ruby (MRI) binding for the libmdbx database library.
|
25
|
+
|
26
|
+
libmdbx is an extremely fast, compact, powerful, embedded, transactional
|
27
|
+
key-value database, with permissive license. libmdbx has a specific set
|
28
|
+
of properties and capabilities, focused on creating unique lightweight
|
29
|
+
solutions.
|
30
|
+
|
31
|
+
- Allows a swarm of multi-threaded processes to ACIDly read and update
|
32
|
+
several key-value maps and multimaps in a locally-shared database.
|
33
|
+
|
34
|
+
- Provides extraordinary performance, minimal overhead through
|
35
|
+
Memory-Mapping and Olog(N) operations costs by virtue of B+ tree.
|
36
|
+
|
37
|
+
- Requires no maintenance and no crash recovery since it doesn't use
|
38
|
+
WAL, but that might be a caveat for write-intensive workloads with
|
39
|
+
durability requirements.
|
40
|
+
|
41
|
+
- Compact and friendly for fully embedding. Only ≈25KLOC of C11,
|
42
|
+
≈64K x86 binary code of core, no internal threads neither server
|
43
|
+
process(es), but implements a simplified variant of the Berkeley DB
|
44
|
+
and dbm API.
|
45
|
+
|
46
|
+
- Enforces serializability for writers just by single mutex and
|
47
|
+
affords wait-free for parallel readers without atomic/interlocked
|
48
|
+
operations, while writing and reading transactions do not block each
|
49
|
+
other.
|
50
|
+
|
51
|
+
- Guarantee data integrity after crash unless this was explicitly
|
52
|
+
neglected in favour of write performance.
|
53
|
+
|
54
|
+
- Supports Linux, Windows, MacOS, Android, iOS, FreeBSD, DragonFly,
|
55
|
+
Solaris, OpenSolaris, OpenIndiana, NetBSD, OpenBSD and other systems
|
56
|
+
compliant with POSIX.1-2008.
|
57
|
+
|
58
|
+
- Historically, libmdbx is a deeply revised and extended descendant
|
59
|
+
of the amazing Lightning Memory-Mapped Database. libmdbx inherits
|
60
|
+
all benefits from LMDB, but resolves some issues and adds a set of
|
61
|
+
improvements.
|
62
|
+
|
63
|
+
|
64
|
+
### Examples
|
65
|
+
|
66
|
+
[forthcoming]
|
67
|
+
|
68
|
+
|
69
|
+
## Prerequisites
|
70
|
+
|
71
|
+
* Ruby 2.6+
|
72
|
+
* libmdbx (https://github.com/erthink/libmdbx)
|
73
|
+
|
74
|
+
|
75
|
+
## Installation
|
76
|
+
|
77
|
+
$ gem install mdbx
|
78
|
+
|
79
|
+
|
80
|
+
## Contributing
|
81
|
+
|
82
|
+
You can check out the current development source with Mercurial via its
|
83
|
+
[home repo](https://code.martini.nu/ruby-mdbx), or with Git at its
|
84
|
+
[project page](https://gitlab.com/mahlon/ruby-mdbx).
|
85
|
+
|
86
|
+
After checking out the source, run:
|
87
|
+
|
88
|
+
$ gem install -Ng
|
89
|
+
$ rake setup
|
90
|
+
|
91
|
+
This will install dependencies, and do any other necessary setup for
|
92
|
+
development.
|
93
|
+
|
94
|
+
|
95
|
+
## Authors
|
96
|
+
|
97
|
+
- Mahlon E. Smith <mahlon@martini.nu>
|
98
|
+
|
99
|
+
|
100
|
+
## License
|
101
|
+
|
102
|
+
Copyright (c) 2020, Mahlon E. Smith
|
103
|
+
All rights reserved.
|
104
|
+
|
105
|
+
Redistribution and use in source and binary forms, with or without
|
106
|
+
modification, are permitted provided that the following conditions are met:
|
107
|
+
|
108
|
+
* Redistributions of source code must retain the above copyright notice,
|
109
|
+
this list of conditions and the following disclaimer.
|
110
|
+
|
111
|
+
* Redistributions in binary form must reproduce the above copyright notice,
|
112
|
+
this list of conditions and the following disclaimer in the documentation
|
113
|
+
and/or other materials provided with the distribution.
|
114
|
+
|
115
|
+
* Neither the name of the author/s, nor the names of the project's
|
116
|
+
contributors may be used to endorse or promote products derived from this
|
117
|
+
software without specific prior written permission.
|
118
|
+
|
119
|
+
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
120
|
+
AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
121
|
+
IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
122
|
+
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
123
|
+
FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
124
|
+
DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
125
|
+
SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
126
|
+
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
127
|
+
OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
128
|
+
OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
129
|
+
|
@@ -0,0 +1,511 @@
|
|
1
|
+
/* vim: set noet sta sw=4 ts=4 : */
|
2
|
+
|
3
|
+
#include "mdbx_ext.h"
|
4
|
+
|
5
|
+
/* VALUE str = rb_sprintf( "path: %+"PRIsVALUE", opts: %+"PRIsVALUE, path, opts ); */
|
6
|
+
/* printf( "%s\n", StringValueCStr(str) ); */
|
7
|
+
|
8
|
+
VALUE rmdbx_cDatabase;
|
9
|
+
|
10
|
+
|
11
|
+
/* Shortcut for fetching current DB variables.
|
12
|
+
*
|
13
|
+
*/
|
14
|
+
#define UNWRAP_DB( val, db ) \
|
15
|
+
rmdbx_db_t *db; \
|
16
|
+
TypedData_Get_Struct( val, rmdbx_db_t, &rmdbx_db_data, db );
|
17
|
+
|
18
|
+
|
19
|
+
/*
|
20
|
+
* A struct encapsulating an instance's DB state.
|
21
|
+
*/
|
22
|
+
struct rmdbx_db {
|
23
|
+
MDBX_env *env;
|
24
|
+
MDBX_dbi dbi;
|
25
|
+
MDBX_txn *txn;
|
26
|
+
MDBX_cursor *cursor;
|
27
|
+
int env_flags;
|
28
|
+
int mode;
|
29
|
+
int open;
|
30
|
+
int max_collections;
|
31
|
+
char *path;
|
32
|
+
char *subdb;
|
33
|
+
};
|
34
|
+
typedef struct rmdbx_db rmdbx_db_t;
|
35
|
+
|
36
|
+
|
37
|
+
/*
|
38
|
+
* Ruby allocation hook.
|
39
|
+
*/
|
40
|
+
void rmdbx_free( void *db ); /* forward declaration */
|
41
|
+
static const rb_data_type_t rmdbx_db_data = {
|
42
|
+
.wrap_struct_name = "MDBX::Database::Data",
|
43
|
+
.function = { .dfree = rmdbx_free },
|
44
|
+
.flags = RUBY_TYPED_FREE_IMMEDIATELY
|
45
|
+
};
|
46
|
+
|
47
|
+
|
48
|
+
/*
|
49
|
+
* Allocate a DB environment onto the stack.
|
50
|
+
*/
|
51
|
+
VALUE
|
52
|
+
rmdbx_alloc( VALUE klass )
|
53
|
+
{
|
54
|
+
rmdbx_db_t *new = RB_ALLOC( rmdbx_db_t );
|
55
|
+
return TypedData_Make_Struct( klass, rmdbx_db_t, &rmdbx_db_data, new );
|
56
|
+
}
|
57
|
+
|
58
|
+
|
59
|
+
/*
|
60
|
+
* Ensure all database file descriptors are collected and
|
61
|
+
* removed.
|
62
|
+
*/
|
63
|
+
void
|
64
|
+
rmdbx_close_all( rmdbx_db_t* db )
|
65
|
+
{
|
66
|
+
if ( db->cursor ) mdbx_cursor_close( db->cursor );
|
67
|
+
if ( db->txn ) mdbx_txn_abort( db->txn );
|
68
|
+
if ( db->dbi ) mdbx_dbi_close( db->env, db->dbi );
|
69
|
+
if ( db->env ) mdbx_env_close( db->env );
|
70
|
+
db->open = 0;
|
71
|
+
}
|
72
|
+
|
73
|
+
|
74
|
+
/*
|
75
|
+
* Cleanup a previously allocated DB environment.
|
76
|
+
*/
|
77
|
+
void
|
78
|
+
rmdbx_free( void *db )
|
79
|
+
{
|
80
|
+
if ( db ) {
|
81
|
+
rmdbx_close_all( db );
|
82
|
+
free( db );
|
83
|
+
}
|
84
|
+
}
|
85
|
+
|
86
|
+
|
87
|
+
/*
|
88
|
+
* Cleanly close an opened database from Ruby.
|
89
|
+
*/
|
90
|
+
VALUE
|
91
|
+
rmdbx_close( VALUE self )
|
92
|
+
{
|
93
|
+
UNWRAP_DB( self, db );
|
94
|
+
rmdbx_close_all( db );
|
95
|
+
return Qtrue;
|
96
|
+
}
|
97
|
+
|
98
|
+
|
99
|
+
/*
|
100
|
+
* Open the DB environment handle.
|
101
|
+
*/
|
102
|
+
VALUE
|
103
|
+
rmdbx_open_env( VALUE self )
|
104
|
+
{
|
105
|
+
int rc;
|
106
|
+
UNWRAP_DB( self, db );
|
107
|
+
rmdbx_close_all( db );
|
108
|
+
|
109
|
+
/* Allocate an mdbx environment.
|
110
|
+
*/
|
111
|
+
rc = mdbx_env_create( &db->env );
|
112
|
+
if ( rc != MDBX_SUCCESS )
|
113
|
+
rb_raise( rmdbx_eDatabaseError, "mdbx_env_create: (%d) %s", rc, mdbx_strerror(rc) );
|
114
|
+
|
115
|
+
/* Set the maximum number of named databases for the environment. */
|
116
|
+
// FIXME: potenially more env setups here? maxreaders, pagesize?
|
117
|
+
mdbx_env_set_maxdbs( db->env, db->max_collections );
|
118
|
+
|
119
|
+
rc = mdbx_env_open( db->env, db->path, db->env_flags, db->mode );
|
120
|
+
if ( rc != MDBX_SUCCESS ) {
|
121
|
+
rmdbx_close( self );
|
122
|
+
rb_raise( rmdbx_eDatabaseError, "mdbx_env_open: (%d) %s", rc, mdbx_strerror(rc) );
|
123
|
+
}
|
124
|
+
db->open = 1;
|
125
|
+
|
126
|
+
return Qtrue;
|
127
|
+
}
|
128
|
+
|
129
|
+
|
130
|
+
/*
|
131
|
+
* call-seq:
|
132
|
+
* db.closed? #=> false
|
133
|
+
*
|
134
|
+
* Predicate: return true if the database environment is closed.
|
135
|
+
*/
|
136
|
+
VALUE
|
137
|
+
rmdbx_closed_p( VALUE self )
|
138
|
+
{
|
139
|
+
UNWRAP_DB( self, db );
|
140
|
+
return db->open == 1 ? Qfalse : Qtrue;
|
141
|
+
}
|
142
|
+
|
143
|
+
|
144
|
+
/*
|
145
|
+
* Open a new database transaction.
|
146
|
+
*
|
147
|
+
* +rwflag+ must be either MDBX_TXN_RDONLY or MDBX_TXN_READWRITE.
|
148
|
+
*/
|
149
|
+
void
|
150
|
+
rmdbx_open_txn( VALUE self, int rwflag )
|
151
|
+
{
|
152
|
+
int rc;
|
153
|
+
UNWRAP_DB( self, db );
|
154
|
+
|
155
|
+
rc = mdbx_txn_begin( db->env, NULL, rwflag, &db->txn);
|
156
|
+
if ( rc != MDBX_SUCCESS ) {
|
157
|
+
rmdbx_close( self );
|
158
|
+
rb_raise( rmdbx_eDatabaseError, "mdbx_txn_begin: (%d) %s", rc, mdbx_strerror(rc) );
|
159
|
+
}
|
160
|
+
|
161
|
+
if ( db->dbi == 0 ) {
|
162
|
+
// FIXME: dbi_flags
|
163
|
+
rc = mdbx_dbi_open( db->txn, db->subdb, MDBX_CREATE, &db->dbi );
|
164
|
+
if ( rc != MDBX_SUCCESS ) {
|
165
|
+
rmdbx_close( self );
|
166
|
+
rb_raise( rmdbx_eDatabaseError, "mdbx_dbi_open: (%d) %s", rc, mdbx_strerror(rc) );
|
167
|
+
}
|
168
|
+
}
|
169
|
+
|
170
|
+
return;
|
171
|
+
}
|
172
|
+
|
173
|
+
|
174
|
+
/*
|
175
|
+
* call-seq:
|
176
|
+
* db.clear
|
177
|
+
*
|
178
|
+
* Empty the database (or collection) on disk. Unrecoverable!
|
179
|
+
*/
|
180
|
+
VALUE
|
181
|
+
rmdbx_clear( VALUE self )
|
182
|
+
{
|
183
|
+
UNWRAP_DB( self, db );
|
184
|
+
|
185
|
+
rmdbx_open_txn( self, MDBX_TXN_READWRITE );
|
186
|
+
int rc = mdbx_drop( db->txn, db->dbi, true );
|
187
|
+
|
188
|
+
if ( rc != MDBX_SUCCESS )
|
189
|
+
rb_raise( rmdbx_eDatabaseError, "mdbx_drop: (%d) %s", rc, mdbx_strerror(rc) );
|
190
|
+
|
191
|
+
mdbx_txn_commit( db->txn );
|
192
|
+
|
193
|
+
/* Refresh the environment handles. */
|
194
|
+
rmdbx_open_env( self );
|
195
|
+
|
196
|
+
return Qnil;
|
197
|
+
}
|
198
|
+
|
199
|
+
|
200
|
+
/*
|
201
|
+
* Given a ruby +arg+, convert and return a structure
|
202
|
+
* suitable for usage as a key for mdbx. All keys are explicitly
|
203
|
+
* converted to strings.
|
204
|
+
*/
|
205
|
+
MDBX_val
|
206
|
+
rmdbx_key_for( VALUE arg )
|
207
|
+
{
|
208
|
+
MDBX_val rv;
|
209
|
+
|
210
|
+
arg = rb_funcall( arg, rb_intern("to_s"), 0 );
|
211
|
+
rv.iov_len = RSTRING_LEN( arg );
|
212
|
+
rv.iov_base = StringValuePtr( arg );
|
213
|
+
|
214
|
+
return rv;
|
215
|
+
}
|
216
|
+
|
217
|
+
|
218
|
+
/*
|
219
|
+
* Given a ruby +arg+, convert and return a structure
|
220
|
+
* suitable for usage as a value for mdbx.
|
221
|
+
*/
|
222
|
+
MDBX_val
|
223
|
+
rmdbx_val_for( VALUE self, VALUE arg )
|
224
|
+
{
|
225
|
+
MDBX_val rv;
|
226
|
+
VALUE serialize_proc;
|
227
|
+
|
228
|
+
serialize_proc = rb_iv_get( self, "@serializer" );
|
229
|
+
if ( ! NIL_P( serialize_proc ) )
|
230
|
+
arg = rb_funcall( serialize_proc, rb_intern("call"), 1, arg );
|
231
|
+
|
232
|
+
rv.iov_len = RSTRING_LEN( arg );
|
233
|
+
rv.iov_base = StringValuePtr( arg );
|
234
|
+
|
235
|
+
return rv;
|
236
|
+
}
|
237
|
+
|
238
|
+
|
239
|
+
/* call-seq:
|
240
|
+
* db.keys #=> [ 'key1', 'key2', ... ]
|
241
|
+
*
|
242
|
+
* Return an array of all keys in the current collection.
|
243
|
+
*/
|
244
|
+
VALUE
|
245
|
+
rmdbx_keys( VALUE self )
|
246
|
+
{
|
247
|
+
UNWRAP_DB( self, db );
|
248
|
+
VALUE rv = rb_ary_new();
|
249
|
+
MDBX_val key, data;
|
250
|
+
int rc;
|
251
|
+
|
252
|
+
if ( ! db->open ) rb_raise( rmdbx_eDatabaseError, "Closed database." );
|
253
|
+
|
254
|
+
rmdbx_open_txn( self, MDBX_TXN_RDONLY );
|
255
|
+
rc = mdbx_cursor_open( db->txn, db->dbi, &db->cursor);
|
256
|
+
|
257
|
+
if ( rc != MDBX_SUCCESS ) {
|
258
|
+
rmdbx_close( self );
|
259
|
+
rb_raise( rmdbx_eDatabaseError, "Unable to open cursor: (%d) %s", rc, mdbx_strerror(rc) );
|
260
|
+
}
|
261
|
+
|
262
|
+
rc = mdbx_cursor_get( db->cursor, &key, &data, MDBX_FIRST );
|
263
|
+
if ( rc == MDBX_SUCCESS ) {
|
264
|
+
rb_ary_push( rv, rb_str_new( key.iov_base, key.iov_len ) );
|
265
|
+
while ( mdbx_cursor_get( db->cursor, &key, &data, MDBX_NEXT ) == 0 ) {
|
266
|
+
rb_ary_push( rv, rb_str_new( key.iov_base, key.iov_len ) );
|
267
|
+
}
|
268
|
+
}
|
269
|
+
|
270
|
+
mdbx_cursor_close( db->cursor );
|
271
|
+
db->cursor = NULL;
|
272
|
+
mdbx_txn_abort( db->txn );
|
273
|
+
return rv;
|
274
|
+
}
|
275
|
+
|
276
|
+
|
277
|
+
/* call-seq:
|
278
|
+
* db[ 'key' ] #=> value
|
279
|
+
*
|
280
|
+
* Convenience method: return a single value for +key+ immediately.
|
281
|
+
*/
|
282
|
+
VALUE
|
283
|
+
rmdbx_get_val( VALUE self, VALUE key )
|
284
|
+
{
|
285
|
+
int rc;
|
286
|
+
VALUE deserialize_proc;
|
287
|
+
UNWRAP_DB( self, db );
|
288
|
+
|
289
|
+
if ( ! db->open ) rb_raise( rmdbx_eDatabaseError, "Closed database." );
|
290
|
+
|
291
|
+
rmdbx_open_txn( self, MDBX_TXN_RDONLY );
|
292
|
+
|
293
|
+
MDBX_val ckey = rmdbx_key_for( key );
|
294
|
+
MDBX_val data;
|
295
|
+
rc = mdbx_get( db->txn, db->dbi, &ckey, &data );
|
296
|
+
mdbx_txn_abort( db->txn );
|
297
|
+
|
298
|
+
switch ( rc ) {
|
299
|
+
case MDBX_SUCCESS:
|
300
|
+
deserialize_proc = rb_iv_get( self, "@deserializer" );
|
301
|
+
VALUE rv = rb_str_new( data.iov_base, data.iov_len );
|
302
|
+
if ( ! NIL_P( deserialize_proc ) )
|
303
|
+
return rb_funcall( deserialize_proc, rb_intern("call"), 1, rv );
|
304
|
+
return rv;
|
305
|
+
|
306
|
+
case MDBX_NOTFOUND:
|
307
|
+
return Qnil;
|
308
|
+
|
309
|
+
default:
|
310
|
+
rmdbx_close( self );
|
311
|
+
rb_raise( rmdbx_eDatabaseError, "Unable to fetch value: (%d) %s", rc, mdbx_strerror(rc) );
|
312
|
+
}
|
313
|
+
}
|
314
|
+
|
315
|
+
|
316
|
+
/* call-seq:
|
317
|
+
* db[ 'key' ] = value #=> value
|
318
|
+
*
|
319
|
+
* Convenience method: set a single value for +key+
|
320
|
+
*/
|
321
|
+
VALUE
|
322
|
+
rmdbx_put_val( VALUE self, VALUE key, VALUE val )
|
323
|
+
{
|
324
|
+
int rc;
|
325
|
+
UNWRAP_DB( self, db );
|
326
|
+
|
327
|
+
if ( ! db->open ) rb_raise( rmdbx_eDatabaseError, "Closed database." );
|
328
|
+
|
329
|
+
rmdbx_open_txn( self, MDBX_TXN_READWRITE );
|
330
|
+
|
331
|
+
MDBX_val ckey = rmdbx_key_for( key );
|
332
|
+
|
333
|
+
// FIXME: DUPSORT is enabled -- different api?
|
334
|
+
// See: MDBX_NODUPDATA / MDBX_NOOVERWRITE
|
335
|
+
if ( NIL_P(val) ) { /* remove if set to nil */
|
336
|
+
rc = mdbx_del( db->txn, db->dbi, &ckey, NULL );
|
337
|
+
}
|
338
|
+
else {
|
339
|
+
MDBX_val old;
|
340
|
+
MDBX_val data = rmdbx_val_for( self, val );
|
341
|
+
rc = mdbx_replace( db->txn, db->dbi, &ckey, &data, &old, 0 );
|
342
|
+
}
|
343
|
+
|
344
|
+
mdbx_txn_commit( db->txn );
|
345
|
+
|
346
|
+
switch ( rc ) {
|
347
|
+
case MDBX_SUCCESS:
|
348
|
+
return val;
|
349
|
+
case MDBX_NOTFOUND:
|
350
|
+
return Qnil;
|
351
|
+
default:
|
352
|
+
rb_raise( rmdbx_eDatabaseError, "Unable to store value: (%d) %s", rc, mdbx_strerror(rc) );
|
353
|
+
}
|
354
|
+
}
|
355
|
+
|
356
|
+
|
357
|
+
/*
|
358
|
+
* call-seq:
|
359
|
+
* db.collection( 'collection_name' ) # => db
|
360
|
+
* db.collection( nil ) # => db (main)
|
361
|
+
*
|
362
|
+
* Operate on a sub-database "collection". Passing +nil+
|
363
|
+
* sets the database to the main, top-level namespace.
|
364
|
+
*
|
365
|
+
*/
|
366
|
+
VALUE
|
367
|
+
rmdbx_set_subdb( int argc, VALUE *argv, VALUE self )
|
368
|
+
{
|
369
|
+
UNWRAP_DB( self, db );
|
370
|
+
VALUE subdb;
|
371
|
+
|
372
|
+
rb_scan_args( argc, argv, "01", &subdb );
|
373
|
+
if ( argc == 0 ) {
|
374
|
+
if ( db->subdb == NULL ) return Qnil;
|
375
|
+
return rb_str_new_cstr( db->subdb );
|
376
|
+
}
|
377
|
+
|
378
|
+
rb_iv_set( self, "@collection", subdb );
|
379
|
+
db->subdb = NIL_P( subdb ) ? NULL : StringValueCStr( subdb );
|
380
|
+
|
381
|
+
/* Close any currently open dbi handle, to be re-opened with
|
382
|
+
* the new collection on next access.
|
383
|
+
*
|
384
|
+
FIXME: Immediate transaction write to auto-create new env?
|
385
|
+
Fetching from here at the moment causes an error if you
|
386
|
+
haven't written anything yet.
|
387
|
+
*/
|
388
|
+
if ( db->dbi ) {
|
389
|
+
mdbx_dbi_close( db->env, db->dbi );
|
390
|
+
db->dbi = 0;
|
391
|
+
}
|
392
|
+
|
393
|
+
return self;
|
394
|
+
}
|
395
|
+
|
396
|
+
|
397
|
+
/*
|
398
|
+
* call-seq:
|
399
|
+
* MDBX::Database.open( path ) -> db
|
400
|
+
* MDBX::Database.open( path, options ) -> db
|
401
|
+
* MDBX::Database.open( path, options ) do |db|
|
402
|
+
* db...
|
403
|
+
* end
|
404
|
+
*
|
405
|
+
* Open an existing (or create a new) mdbx database at filesystem
|
406
|
+
* +path+. In block form, the database is automatically closed.
|
407
|
+
*
|
408
|
+
*/
|
409
|
+
VALUE
|
410
|
+
rmdbx_database_initialize( int argc, VALUE *argv, VALUE self )
|
411
|
+
{
|
412
|
+
int mode = 0644;
|
413
|
+
int max_collections = 0;
|
414
|
+
int env_flags = MDBX_ENV_DEFAULTS;
|
415
|
+
VALUE path, opts, opt;
|
416
|
+
|
417
|
+
rb_scan_args( argc, argv, "11", &path, &opts );
|
418
|
+
|
419
|
+
/* Ensure options is a hash if it was passed in.
|
420
|
+
*/
|
421
|
+
if ( NIL_P(opts) ) {
|
422
|
+
opts = rb_hash_new();
|
423
|
+
}
|
424
|
+
else {
|
425
|
+
Check_Type( opts, T_HASH );
|
426
|
+
}
|
427
|
+
rb_hash_freeze( opts );
|
428
|
+
|
429
|
+
/* Options setup, overrides.
|
430
|
+
*/
|
431
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("mode") ) );
|
432
|
+
if ( ! NIL_P(opt) ) mode = FIX2INT( opt );
|
433
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("max_collections") ) );
|
434
|
+
if ( ! NIL_P(opt) ) max_collections = FIX2INT( opt );
|
435
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("nosubdir") ) );
|
436
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_NOSUBDIR;
|
437
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("readonly") ) );
|
438
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_RDONLY;
|
439
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("exclusive") ) );
|
440
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_EXCLUSIVE;
|
441
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("compat") ) );
|
442
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_ACCEDE;
|
443
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("writemap") ) );
|
444
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_WRITEMAP;
|
445
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("no_threadlocal") ) );
|
446
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_NOTLS;
|
447
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("no_readahead") ) );
|
448
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_NORDAHEAD;
|
449
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("no_memory_init") ) );
|
450
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_NOMEMINIT;
|
451
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("coalesce") ) );
|
452
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_COALESCE;
|
453
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("lifo_reclaim") ) );
|
454
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_LIFORECLAIM;
|
455
|
+
opt = rb_hash_aref( opts, ID2SYM( rb_intern("no_metasync") ) );
|
456
|
+
if ( RTEST(opt) ) env_flags = env_flags | MDBX_NOMETASYNC;
|
457
|
+
|
458
|
+
/* Duplicate keys, on mdbx_dbi_open, maybe set here? */
|
459
|
+
/* MDBX_DUPSORT = UINT32_C(0x04), */
|
460
|
+
|
461
|
+
/* Initialize the DB vals.
|
462
|
+
*/
|
463
|
+
UNWRAP_DB( self, db );
|
464
|
+
db->env = NULL;
|
465
|
+
db->dbi = 0;
|
466
|
+
db->txn = NULL;
|
467
|
+
db->cursor = NULL;
|
468
|
+
db->env_flags = env_flags;
|
469
|
+
db->mode = mode;
|
470
|
+
db->max_collections = max_collections;
|
471
|
+
db->path = StringValueCStr( path );
|
472
|
+
db->open = 0;
|
473
|
+
db->subdb = NULL;
|
474
|
+
|
475
|
+
/* Set instance variables.
|
476
|
+
*/
|
477
|
+
rb_iv_set( self, "@path", path );
|
478
|
+
rb_iv_set( self, "@options", opts );
|
479
|
+
|
480
|
+
rmdbx_open_env( self );
|
481
|
+
return self;
|
482
|
+
}
|
483
|
+
|
484
|
+
|
485
|
+
/*
|
486
|
+
* Initialization for the MDBX::Database class.
|
487
|
+
*/
|
488
|
+
void
|
489
|
+
rmdbx_init_database()
|
490
|
+
{
|
491
|
+
rmdbx_cDatabase = rb_define_class_under( rmdbx_mMDBX, "Database", rb_cData );
|
492
|
+
|
493
|
+
#ifdef FOR_RDOC
|
494
|
+
rmdbx_mMDBX = rb_define_module( "MDBX" );
|
495
|
+
#endif
|
496
|
+
|
497
|
+
rb_define_alloc_func( rmdbx_cDatabase, rmdbx_alloc );
|
498
|
+
|
499
|
+
rb_define_protected_method( rmdbx_cDatabase, "initialize", rmdbx_database_initialize, -1 );
|
500
|
+
rb_define_method( rmdbx_cDatabase, "collection", rmdbx_set_subdb, -1 );
|
501
|
+
rb_define_method( rmdbx_cDatabase, "close", rmdbx_close, 0 );
|
502
|
+
rb_define_method( rmdbx_cDatabase, "reopen", rmdbx_open_env, 0 );
|
503
|
+
rb_define_method( rmdbx_cDatabase, "closed?", rmdbx_closed_p, 0 );
|
504
|
+
rb_define_method( rmdbx_cDatabase, "clear", rmdbx_clear, 0 );
|
505
|
+
rb_define_method( rmdbx_cDatabase, "keys", rmdbx_keys, 0 );
|
506
|
+
rb_define_method( rmdbx_cDatabase, "[]", rmdbx_get_val, 1 );
|
507
|
+
rb_define_method( rmdbx_cDatabase, "[]=", rmdbx_put_val, 2 );
|
508
|
+
|
509
|
+
rb_require( "mdbx/database" );
|
510
|
+
}
|
511
|
+
|
@@ -0,0 +1,26 @@
|
|
1
|
+
/* vim: set noet sta sw=4 ts=4 : */
|
2
|
+
|
3
|
+
#include "mdbx_ext.h"
|
4
|
+
|
5
|
+
VALUE rmdbx_mMDBX;
|
6
|
+
VALUE rmdbx_eDatabaseError;
|
7
|
+
VALUE rmdbx_eRollback;
|
8
|
+
|
9
|
+
/*
|
10
|
+
* MDBX initialization
|
11
|
+
*/
|
12
|
+
void
|
13
|
+
Init_mdbx_ext()
|
14
|
+
{
|
15
|
+
rmdbx_mMDBX = rb_define_module( "MDBX" );
|
16
|
+
|
17
|
+
/* The backend library version. */
|
18
|
+
VALUE version = rb_str_new_cstr( mdbx_version.git.describe );
|
19
|
+
rb_define_const( rmdbx_mMDBX, "LIBRARY_VERSION", version );
|
20
|
+
|
21
|
+
rmdbx_eDatabaseError = rb_define_class_under( rmdbx_mMDBX, "DatabaseError", rb_eRuntimeError );
|
22
|
+
rmdbx_eRollback = rb_define_class_under( rmdbx_mMDBX, "Rollback", rb_eRuntimeError );
|
23
|
+
|
24
|
+
rmdbx_init_database();
|
25
|
+
}
|
26
|
+
|
@@ -0,0 +1,29 @@
|
|
1
|
+
|
2
|
+
#include <ruby.h>
|
3
|
+
#include "extconf.h"
|
4
|
+
|
5
|
+
#include "mdbx.h"
|
6
|
+
|
7
|
+
#ifndef MDBX_EXT_0_9_2
|
8
|
+
#define MDBX_EXT_0_9_2
|
9
|
+
|
10
|
+
|
11
|
+
/* ------------------------------------------------------------
|
12
|
+
* Globals
|
13
|
+
* ------------------------------------------------------------ */
|
14
|
+
|
15
|
+
extern VALUE rmdbx_mMDBX;
|
16
|
+
extern VALUE rmdbx_cDatabase;
|
17
|
+
extern VALUE rmdbx_eDatabaseError;
|
18
|
+
extern VALUE rmdbx_eRollback;
|
19
|
+
|
20
|
+
|
21
|
+
/* ------------------------------------------------------------
|
22
|
+
* Functions
|
23
|
+
* ------------------------------------------------------------ */
|
24
|
+
extern void Init_rmdbx ( void );
|
25
|
+
extern void rmdbx_init_database ( void );
|
26
|
+
|
27
|
+
|
28
|
+
#endif /* define MDBX_EXT_0_9_2 */
|
29
|
+
|
data/lib/mdbx.rb
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
# -*- ruby -*-
|
2
|
+
# vim: set nosta noet ts=4 sw=4 ft=ruby:
|
3
|
+
# encoding: utf-8
|
4
|
+
|
5
|
+
require 'mdbx_ext'
|
6
|
+
|
7
|
+
|
8
|
+
# Top level namespace for MDBX.
|
9
|
+
#
|
10
|
+
module MDBX
|
11
|
+
|
12
|
+
# The version of this gem.
|
13
|
+
#
|
14
|
+
# Note: the MDBX library version this gem was built
|
15
|
+
# against can be found in the 'LIBRARY_VERSION' constant.
|
16
|
+
#
|
17
|
+
VERSION = '0.0.1'
|
18
|
+
|
19
|
+
end # module MDBX
|
20
|
+
|
@@ -0,0 +1,68 @@
|
|
1
|
+
# -*- ruby -*-
|
2
|
+
# vim: set nosta noet ts=4 sw=4 ft=ruby:
|
3
|
+
# encoding: utf-8
|
4
|
+
|
5
|
+
require 'mdbx' unless defined?( MDBX )
|
6
|
+
|
7
|
+
|
8
|
+
# TODO: rdoc
|
9
|
+
#
|
10
|
+
class MDBX::Database
|
11
|
+
|
12
|
+
### Open an existing (or create a new) mdbx database at filesystem
|
13
|
+
### +path+. In block form, the database is automatically closed.
|
14
|
+
###
|
15
|
+
### MDBX::Database.open( path ) -> db
|
16
|
+
### MDBX::Database.open( path, options ) -> db
|
17
|
+
### MDBX::Database.open( path, options ) do |db|
|
18
|
+
### db[ 'key' ] #=> value
|
19
|
+
### end
|
20
|
+
###
|
21
|
+
### FIXME: options!
|
22
|
+
###
|
23
|
+
def self::open( *args, &block )
|
24
|
+
db = new( *args )
|
25
|
+
|
26
|
+
db.serializer = ->( v ) { Marshal.dump( v ) }
|
27
|
+
db.deserializer = ->( v ) { Marshal.load( v ) }
|
28
|
+
|
29
|
+
if block_given?
|
30
|
+
begin
|
31
|
+
yield db
|
32
|
+
ensure
|
33
|
+
db.close
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
return db
|
38
|
+
end
|
39
|
+
|
40
|
+
|
41
|
+
# Only instantiate Database objects via #open.
|
42
|
+
private_class_method :new
|
43
|
+
|
44
|
+
|
45
|
+
# The options used to instantiate this database.
|
46
|
+
attr_reader :options
|
47
|
+
|
48
|
+
# The path on disk of the database.
|
49
|
+
attr_reader :path
|
50
|
+
|
51
|
+
# A Proc for automatically serializing values.
|
52
|
+
attr_accessor :serializer
|
53
|
+
|
54
|
+
# A Proc for automatically deserializing values.
|
55
|
+
attr_accessor :deserializer
|
56
|
+
|
57
|
+
|
58
|
+
### Switch to the top-level collection.
|
59
|
+
###
|
60
|
+
def main
|
61
|
+
return self.collection( nil )
|
62
|
+
end
|
63
|
+
|
64
|
+
# Allow for some common nomenclature.
|
65
|
+
alias_method :namespace, :collection
|
66
|
+
|
67
|
+
end # class MDBX::Database
|
68
|
+
|
data/lib/mdbx_ext.so
ADDED
Binary file
|
metadata
ADDED
@@ -0,0 +1,208 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: mdbx
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.1.0.pre.20201217111933
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Mahlon E. Smith
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain:
|
11
|
+
- |
|
12
|
+
-----BEGIN CERTIFICATE-----
|
13
|
+
MIIENDCCApygAwIBAgIBATANBgkqhkiG9w0BAQsFADAiMSAwHgYDVQQDDBdtYWhs
|
14
|
+
b24vREM9bWFydGluaS9EQz1udTAeFw0yMDEyMjIwMDQwNDBaFw0zMDEyMjAwMDQw
|
15
|
+
NDBaMCIxIDAeBgNVBAMMF21haGxvbi9EQz1tYXJ0aW5pL0RDPW51MIIBojANBgkq
|
16
|
+
hkiG9w0BAQEFAAOCAY8AMIIBigKCAYEA2aRhRIb+Cb3OX58q1VHxmeeFM0R0Tyvh
|
17
|
+
rSRequJbgH05Lph3yPqm9Gdvcvqb+5HE4LdTTdRiQuDGE9y1lxaw7CpNxFHgEqDD
|
18
|
+
iBFDClzExQIosju9hzJv7VVEgBobd4da3I37tQH4bJEweOrpoiPESkezGyyeFxVU
|
19
|
+
Sk4ncvMdzk3/XVhiILln5d+s2kB9H6gKE7UTToUay/Faeoo8Ob6TvYfDmVmRR4Mg
|
20
|
+
AJ/SZs1+XY9FrVesyiU81qVo0k7SfA4nsXAOGeJZ7Fku9lG2EoGAepVRwthMu6nb
|
21
|
+
L+vgc2Yc8FE3gLdxgnnqXyuWy7ivK3RmqXkGq0JR3O3+lXedan9eHVEQjIJTio+i
|
22
|
+
QTo9dwtfMKE8EjzT8uN025xe/PJ8XZOtc3AkO5rGVkRcseuZRL8C6v7Ypr4jA454
|
23
|
+
ZGvaNmaKzLlKl6b7caPaeb+AAeVIyBm3CLVdlUdqibttUrm/N+LmeJA+enp8VYTA
|
24
|
+
g8yHOp2C/uyUHLAwmBt41tZqO6hFmMilAgMBAAGjdTBzMAkGA1UdEwQCMAAwCwYD
|
25
|
+
VR0PBAQDAgSwMB0GA1UdDgQWBBRLTYPDE5cpv/oQ3PVlpuCuDXs5uDAcBgNVHREE
|
26
|
+
FTATgRFtYWhsb25AbWFydGluaS5udTAcBgNVHRIEFTATgRFtYWhsb25AbWFydGlu
|
27
|
+
aS5udTANBgkqhkiG9w0BAQsFAAOCAYEAAKlzY37tbOIEGy/gAu5Z6Y4+u6ph2p0C
|
28
|
+
SeveRaItIYHHk870qpFZEpiYah6iupPGMmOXV3uAYNywQb6MGg/3ZguTIF8fQdJh
|
29
|
+
zb3ltiUNsuCCURNsaJ5XpmifO5xoyM3jqb+LuDUnCuDL1qfM/JNXrBguFOfE1Pgb
|
30
|
+
Hv4Asf3a9o7P9V7lBSr1VAF9UTIEqcsV9ZouxTdP9tHtzUpUMaGXQP1myVUJSE35
|
31
|
+
cwSRJbCf8E0NX0iH31nhboVZODZgwgYiyjv4sbjDgI/1V8q7EaI9IdDWsoi8fm4o
|
32
|
+
KdDjQCJ3z7nUXe0ojb0v9xikanWH2S70UX46A/awTwLCh1nXbRc1SwSzfPWVUZpv
|
33
|
+
f46eTlAa9H9cl8GC1tHzl3NUxQ5/bZcSw8FzJXjGivhsyYJ+rL53Skefy5Jdn7Vs
|
34
|
+
AWgz1mmbCTaQJS+NuCrrtq0NbfgluSB/fGvf9ADVPi9ImWpPY3BQE0s6TmJgBI74
|
35
|
+
/KwqC+7jPv0TJhph1OPgnOvSM/sflGsi
|
36
|
+
-----END CERTIFICATE-----
|
37
|
+
date: 2020-12-17 00:00:00.000000000 Z
|
38
|
+
dependencies:
|
39
|
+
- !ruby/object:Gem::Dependency
|
40
|
+
name: pry
|
41
|
+
requirement: !ruby/object:Gem::Requirement
|
42
|
+
requirements:
|
43
|
+
- - "~>"
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: '0.13'
|
46
|
+
type: :development
|
47
|
+
prerelease: false
|
48
|
+
version_requirements: !ruby/object:Gem::Requirement
|
49
|
+
requirements:
|
50
|
+
- - "~>"
|
51
|
+
- !ruby/object:Gem::Version
|
52
|
+
version: '0.13'
|
53
|
+
- !ruby/object:Gem::Dependency
|
54
|
+
name: rake
|
55
|
+
requirement: !ruby/object:Gem::Requirement
|
56
|
+
requirements:
|
57
|
+
- - "~>"
|
58
|
+
- !ruby/object:Gem::Version
|
59
|
+
version: '13.0'
|
60
|
+
type: :development
|
61
|
+
prerelease: false
|
62
|
+
version_requirements: !ruby/object:Gem::Requirement
|
63
|
+
requirements:
|
64
|
+
- - "~>"
|
65
|
+
- !ruby/object:Gem::Version
|
66
|
+
version: '13.0'
|
67
|
+
- !ruby/object:Gem::Dependency
|
68
|
+
name: rake-compiler
|
69
|
+
requirement: !ruby/object:Gem::Requirement
|
70
|
+
requirements:
|
71
|
+
- - "~>"
|
72
|
+
- !ruby/object:Gem::Version
|
73
|
+
version: '1.1'
|
74
|
+
type: :development
|
75
|
+
prerelease: false
|
76
|
+
version_requirements: !ruby/object:Gem::Requirement
|
77
|
+
requirements:
|
78
|
+
- - "~>"
|
79
|
+
- !ruby/object:Gem::Version
|
80
|
+
version: '1.1'
|
81
|
+
- !ruby/object:Gem::Dependency
|
82
|
+
name: rake-deveiate
|
83
|
+
requirement: !ruby/object:Gem::Requirement
|
84
|
+
requirements:
|
85
|
+
- - "~>"
|
86
|
+
- !ruby/object:Gem::Version
|
87
|
+
version: '0.15'
|
88
|
+
- - ">="
|
89
|
+
- !ruby/object:Gem::Version
|
90
|
+
version: 0.15.1
|
91
|
+
type: :development
|
92
|
+
prerelease: false
|
93
|
+
version_requirements: !ruby/object:Gem::Requirement
|
94
|
+
requirements:
|
95
|
+
- - "~>"
|
96
|
+
- !ruby/object:Gem::Version
|
97
|
+
version: '0.15'
|
98
|
+
- - ">="
|
99
|
+
- !ruby/object:Gem::Version
|
100
|
+
version: 0.15.1
|
101
|
+
- !ruby/object:Gem::Dependency
|
102
|
+
name: rdoc-generator-fivefish
|
103
|
+
requirement: !ruby/object:Gem::Requirement
|
104
|
+
requirements:
|
105
|
+
- - "~>"
|
106
|
+
- !ruby/object:Gem::Version
|
107
|
+
version: '0.4'
|
108
|
+
type: :development
|
109
|
+
prerelease: false
|
110
|
+
version_requirements: !ruby/object:Gem::Requirement
|
111
|
+
requirements:
|
112
|
+
- - "~>"
|
113
|
+
- !ruby/object:Gem::Version
|
114
|
+
version: '0.4'
|
115
|
+
- !ruby/object:Gem::Dependency
|
116
|
+
name: rspec
|
117
|
+
requirement: !ruby/object:Gem::Requirement
|
118
|
+
requirements:
|
119
|
+
- - "~>"
|
120
|
+
- !ruby/object:Gem::Version
|
121
|
+
version: '3.9'
|
122
|
+
type: :development
|
123
|
+
prerelease: false
|
124
|
+
version_requirements: !ruby/object:Gem::Requirement
|
125
|
+
requirements:
|
126
|
+
- - "~>"
|
127
|
+
- !ruby/object:Gem::Version
|
128
|
+
version: '3.9'
|
129
|
+
- !ruby/object:Gem::Dependency
|
130
|
+
name: rubocop
|
131
|
+
requirement: !ruby/object:Gem::Requirement
|
132
|
+
requirements:
|
133
|
+
- - "~>"
|
134
|
+
- !ruby/object:Gem::Version
|
135
|
+
version: '0.93'
|
136
|
+
type: :development
|
137
|
+
prerelease: false
|
138
|
+
version_requirements: !ruby/object:Gem::Requirement
|
139
|
+
requirements:
|
140
|
+
- - "~>"
|
141
|
+
- !ruby/object:Gem::Version
|
142
|
+
version: '0.93'
|
143
|
+
- !ruby/object:Gem::Dependency
|
144
|
+
name: simplecov
|
145
|
+
requirement: !ruby/object:Gem::Requirement
|
146
|
+
requirements:
|
147
|
+
- - "~>"
|
148
|
+
- !ruby/object:Gem::Version
|
149
|
+
version: '0.12'
|
150
|
+
type: :development
|
151
|
+
prerelease: false
|
152
|
+
version_requirements: !ruby/object:Gem::Requirement
|
153
|
+
requirements:
|
154
|
+
- - "~>"
|
155
|
+
- !ruby/object:Gem::Version
|
156
|
+
version: '0.12'
|
157
|
+
description: |
|
158
|
+
This is a native ruby binding to libmdbx, an improved version
|
159
|
+
of the Lightning Memory Mapped Database.
|
160
|
+
|
161
|
+
libmdbx is an extremely fast, compact, powerful, embedded,
|
162
|
+
transactional key-value database, with permissive license.
|
163
|
+
libmdbx has a specific set of properties and capabilities,
|
164
|
+
focused on creating unique lightweight solutions.
|
165
|
+
email:
|
166
|
+
- mahlon@martini.nu
|
167
|
+
executables: []
|
168
|
+
extensions:
|
169
|
+
- ext/mdbx_ext/extconf.rb
|
170
|
+
extra_rdoc_files: []
|
171
|
+
files:
|
172
|
+
- History.md
|
173
|
+
- README.md
|
174
|
+
- ext/mdbx_ext/database.c
|
175
|
+
- ext/mdbx_ext/extconf.rb
|
176
|
+
- ext/mdbx_ext/mdbx_ext.c
|
177
|
+
- ext/mdbx_ext/mdbx_ext.h
|
178
|
+
- lib/mdbx.rb
|
179
|
+
- lib/mdbx/database.rb
|
180
|
+
- lib/mdbx_ext.so
|
181
|
+
homepage: https://code.martini.nu/ruby-mdbx
|
182
|
+
licenses:
|
183
|
+
- BSD-3-Clause
|
184
|
+
metadata:
|
185
|
+
changelog_uri: https://martini.nu/docs/ruby-mdbx/History_md.html
|
186
|
+
documentation_uri: https://martini.nu/docs/ruby-mdbx
|
187
|
+
homepage_uri: https://code.martini.nu/ruby-mdbx
|
188
|
+
post_install_message:
|
189
|
+
rdoc_options: []
|
190
|
+
require_paths:
|
191
|
+
- lib
|
192
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
193
|
+
requirements:
|
194
|
+
- - ">="
|
195
|
+
- !ruby/object:Gem::Version
|
196
|
+
version: '0'
|
197
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
198
|
+
requirements:
|
199
|
+
- - ">"
|
200
|
+
- !ruby/object:Gem::Version
|
201
|
+
version: 1.3.1
|
202
|
+
requirements: []
|
203
|
+
rubygems_version: 3.1.4
|
204
|
+
signing_key:
|
205
|
+
specification_version: 4
|
206
|
+
summary: A ruby binding to libmdbx, an improved version of the Lightning Memory Mapped
|
207
|
+
Database.
|
208
|
+
test_files: []
|
metadata.gz.sig
ADDED
@@ -0,0 +1,2 @@
|
|
1
|
+
^/Ʌ>�e��j��LKlē�%�2$��[Y�~��!6d��O�t�0K*�&�B熣��NZ�k����>��sa��pdq�']Z��-��E��K�����P�ǚ3����<����>����{�:��PT���\�-E�n>�[V4��{^���yh?T�+����hRu��N�\�@���儈aV�Sܽ�X��y?�/�QFR�ɣ"���؛�`/���x��j��L��mpdS�wпDq�T������c"���$���kD�W���;(�<:��.���Y +�E ���F�Óɣ_�px�4.+�% z�*���9
|
2
|
+
E��Uh�&Y�?]zȷ�Z<�FKa��֬��*����72�dn��`%
|