Safe Rust code for creating Erlang NIF to store big data
```shell
$ make ct
sh crates/buildcrates.sh clippy
Finished dev [unoptimized + debuginfo] target(s) in 0.02s
sh crates/buildcrates.sh test
Finished test [unoptimized + debuginfo] target(s) in 0.02s
Running unittests (crates/bigdata/target/debug/deps/bigdata-f3da29e6da47f8c3)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running tests/big_data_test.rs (crates/big_data/target/debug/deps/big_data_test-f8d535b5018e2ef3)
running 14 tests test clear ... ok test get ... ok test gettimeindex ... ok test getrowids ... ok test getrange ... ok test getrangerowids ... ok test insert ... ok test lenrangerowids ... ok test lenrowids ... ok test remove ... ok test tolist ... ok test removerowids ... ok test updatecounter ... ok test updateelem ... ok
test result: ok. 14 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Doc-tests big_data
running 12 tests test src/bigdata.rs - bigdata::BigData::get (line 387) ... ok test src/bigdata.rs - bigdata::BigData::insert (line 90) ... ok test src/bigdata.rs - bigdata::BigData::lenrangerowids (line 192) ... ok test src/bigdata.rs - bigdata::BigData::gettimeindex (line 405) ... ok test src/bigdata.rs - bigdata::BigData::getrange (line 464) ... ok test src/bigdata.rs - bigdata::BigData::clear (line 129) ... ok test src/bigdata.rs - bigdata::BigData::getrangerowids (line 491) ... ok test src/bigdata.rs - bigdata::BigData::getrowids (line 429) ... ok test src/bigdata.rs - bigdata::BigData::lenrowids (line 165) ... ok test src/bigdata.rs - bigdata::BigData::remove (line 556) ... ok test src/bigdata.rs - bigdata::BigData::removerowids (line 591) ... ok test src/bigdata.rs - bigdata::BigData::tolist (line 526) ... ok
test result: ok. 12 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 1.51s
./rebar3 do ct --dir test/ct -v --config test/ct/ct.config --sysconfig config/test.config ===> Verifying dependencies... Finished release [optimized] target(s) in 0.02s ===> Analyzing applications... ===> Compiling bigdata_nif ===> Running Common Test suites...
Common Test starting (cwd is /Users/admin/proj/rust/bigdatanif)
CWD set to: "/Users/admin/proj/rust/bigdatanif/build/test/logs/ctrun.nonode@nohost.2021-05-13_15.54.32"
TEST INFO: 1 test(s), 11 case(s) in 1 suite(s)
Testing test.ct: Starting test, 11 test cases %%% bigdataSUITE: ........... Testing test.ct: TEST COMPLETE, 11 ok, 0 failed of 11 test cases
Updating /Users/admin/proj/rust/bigdatanif/build/test/logs/index.html ... done Updating /Users/admin/proj/rust/bigdatanif/build/test/logs/all_runs.html ... done $ make shell
./rebar3 as test shell ===> Verifying dependencies... Finished release [optimized] target(s) in 0.04s ===> Analyzing applications... ===> Compiling bigdatanif Erlang/OTP 22 [erts-10.7.2.1] [source] [64-bit] [smp:8:8] [ds:8:8:10] [async-threads:1] [hipe]
Eshell V10.7.2.1 (abort with ^G) 1> ===> Booted bigdatanif
```
erlang
%% create a player bucket
3> {ok, Ref} = big_data_nif:new().
{ok,#Ref<0.2349964349.40239108.14596>}
%% insert a row to bucket
4> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"1">>,{a,1},1}).
ok
%% query all data of player bucket
5> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"1">>,{a,1},1}]
%% query a row
6> big_data_nif:get_row(Ref,<<"player">>, <<"1">>).
{row_data,<<"1">>,{a,1},1}
%% insert other row
7> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"2">>,{a,1},1}).
ok
8> big_data_nif:get_row(Ref,<<"player">>, <<"2">>).
{row_data,<<"2">>,{a,1},1}
9> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"1">>,{a,1},1},{row_data,<<"2">>,{a,1},1}]
```erlang
1> bigdatanif:get(Ref,<<"player">>).
[]
12>
12> bigdatanif:insert(Ref, <<"player">>, {rowdata,<<"2">>,{a,1},1}).
ok
13> bigdatanif:insert(Ref, <<"player">>, {rowdata,<<"1">>,{a,1},10}).
ok
14> bigdatanif:insert(Ref, <<"player">>, {rowdata,<<"3">>,{a,1},0}).
ok
%% query all data which will be sorted by time
15> bigdatanif:get(Ref,<<"player">>).
[{rowdata,<<"3">>,{a,1},0},
{rowdata,<<"2">>,{a,1},1},
{rowdata,<<"1">>,{a,1},10}]
16> bigdatanif:getrange(Ref,<<"player">>,0,1).
[{rowdata,<<"3">>,{a,1},0},{rowdata,<<"2">>,{a,1},1}]
17> bigdatanif:getrange(Ref,<<"player">>,0,0).
[{rowdata,<<"3">>,{a,1},0}]
18> bigdatanif:getrange(Ref,<<"player">>,0,10).
[{rowdata,<<"3">>,{a,1},0},
{rowdata,<<"2">>,{a,1},1},
{row_data,<<"1">>,{a,1},10}]
%% query the rowids by range time %% included 0 and included 10 %% result will be sorted by time 19> bigdatanif:getrangerowids(Ref,<<"player">>,0,10). [<<"3">>,<<"2">>,<<"1">>] %% query the time by rowid 22> bigdatanif:gettime_index(Ref,<<"player">>,<<"1">>). 10 ```
erlang
33> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1,{a,b},<<"hello">>},0}).
ok
34> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"3">>,{a,1,{a,b},<<"hello">>},0}]
35> big_data_nif:lookup_elem(Ref,<<"player">>,<<"3">>,0).
{a}
36> big_data_nif:lookup_elem(Ref,<<"player">>,<<"3">>,[0,1]).
{a,1}
37> big_data_nif:lookup_elem(Ref,<<"player">>,<<"3">>,[0,1,3]).
{a,1,<<"hello">>}
erlang
24> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1},0}).
ok
25> big_data_nif:update_elem(Ref,<<"player">>,<<"3">>,[{0,b},{1,2}]).
[true,true]
26> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"3">>,{b,2},0}]
erlang
28> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1,3,{4,5}},0}).
ok
29> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"3">>,{a,1,3,{4,5}},0}]
30> big_data_nif:update_counter(Ref,<<"player">>,<<"3">>,[{1,1},{2,5}]).
[true,true]
31> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"3">>,{a,2,8,{4,5}},0}]
erlang
2> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1,3,{4,5}},0}).
ok
3> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"4">>,{a,1,3,{4,5}},0}).
4> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"3">>,{a,1,3,{4,5}},0},
{row_data,<<"4">>,{a,1,3,{4,5}},0}]
%% remove player bucket
6> big_data_nif:remove(Ref,<<"player">>).
ok
7> big_data_nif:get(Ref,<<"player">>).
[]
8> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"4">>,{a,1,3,{4,5}},0}).
ok
9> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1,3,{4,5}},0}).
ok
%% remove a row which belong to player bucket
10> big_data_nif:remove_row(Ref,<<"player">>,<<"3">>).
ok
11> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"4">>,{a,1,3,{4,5}},0}]
12> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1,3,{4,5}},0}).
ok
%% remove all rows which time between 0 and 0
13> big_data_nif:remove_row_ids(Ref,<<"player">>,0,0).
ok
14> big_data_nif:get(Ref,<<"player">>).
[]
15> big_data_nif:insert(Ref, <<"player">>, {row_data,<<"3">>,{a,1,3,{4,5}},0}).
ok
16> big_data_nif:insert(Ref, <<"player1">>, {row_data,<<"4">>,{a,1,3,{4,5}},0}).
ok
17> big_data_nif:get(Ref,<<"player1">>).
[{row_data,<<"4">>,{a,1,3,{4,5}},0}]
18> big_data_nif:get(Ref,<<"player">>).
[{row_data,<<"3">>,{a,1,3,{4,5}},0}]
%% clear all buckets
19> big_data_nif:clear(Ref).
ok
20> big_data_nif:get(Ref,<<"player1">>).
[]
21> big_data_nif:get(Ref,<<"player">>).
[]
```shell $ sh crates/buildcrates.sh bench Finished bench [optimized] target(s) in 0.02s Running unittests (crates/bigdata/target/release/deps/big_data-a47418a78ef79a70)
running 0 tests
test result: ok. 0 passed; 0 failed; 0 ignored; 0 measured; 0 filtered out; finished in 0.00s
Running unittests (crates/big_data/target/release/deps/bench-2cb028b9ea72486c)
running 2 tests test get ... bench: 216 ns/iter (+/- 133) test insert ... bench: 1,160 ns/iter (+/- 141)
test result: ok. 0 passed; 0 failed; 0 ignored; 2 measured ```