Compare commits

..

No commits in common. "4e90e752eb2505b72de97d10653a8f83bcb19912" and "eb3e8b8c1986b7d8709912d38f14e1f5f6d508f8" have entirely different histories.

6 changed files with 89 additions and 193 deletions

View File

@ -24,7 +24,7 @@ NAN_MODULE_INIT(InitAddon)
tpl = Nan::New<v8::FunctionTemplate>(NodeVitastorImage::Create); tpl = Nan::New<v8::FunctionTemplate>(NodeVitastorImage::Create);
tpl->SetClassName(Nan::New("Image").ToLocalChecked()); tpl->SetClassName(Nan::New("Image").ToLocalChecked());
tpl->InstanceTemplate()->SetInternalFieldCount(1); tpl->InstanceTemplate()->SetInternalFieldCount(2);
Nan::SetPrototypeMethod(tpl, "read", NodeVitastorImage::Read); Nan::SetPrototypeMethod(tpl, "read", NodeVitastorImage::Read);
Nan::SetPrototypeMethod(tpl, "write", NodeVitastorImage::Write); Nan::SetPrototypeMethod(tpl, "write", NodeVitastorImage::Write);
@ -67,7 +67,7 @@ NAN_MODULE_INIT(InitAddon)
tpl = Nan::New<v8::FunctionTemplate>(NodeVitastorKVListing::Create); tpl = Nan::New<v8::FunctionTemplate>(NodeVitastorKVListing::Create);
tpl->SetClassName(Nan::New("KVListing").ToLocalChecked()); tpl->SetClassName(Nan::New("KVListing").ToLocalChecked());
tpl->InstanceTemplate()->SetInternalFieldCount(1); tpl->InstanceTemplate()->SetInternalFieldCount(2);
Nan::SetPrototypeMethod(tpl, "next", NodeVitastorKVListing::Next); Nan::SetPrototypeMethod(tpl, "next", NodeVitastorKVListing::Next);
Nan::SetPrototypeMethod(tpl, "close", NodeVitastorKVListing::Close); Nan::SetPrototypeMethod(tpl, "close", NodeVitastorKVListing::Close);

View File

@ -13,8 +13,7 @@
'<!(pkg-config --cflags vitastor)' '<!(pkg-config --cflags vitastor)'
], ],
'libraries': [ 'libraries': [
'<!(pkg-config --libs vitastor)', '<!(pkg-config --libs vitastor)'
'-lvitastor_kv'
] ]
} }
] ]

View File

@ -18,15 +18,13 @@
class NodeVitastorRequest: public Nan::AsyncResource class NodeVitastorRequest: public Nan::AsyncResource
{ {
public: public:
NodeVitastorRequest(NodeVitastor *cli, v8::Local<v8::Function> cb): Nan::AsyncResource("NodeVitastorRequest") NodeVitastorRequest(v8::Local<v8::Function> cb): Nan::AsyncResource("NodeVitastorRequest")
{ {
this->cli = cli;
callback.Reset(cb); callback.Reset(cb);
} }
iovec iov; iovec iov;
std::vector<iovec> iov_list; std::vector<iovec> iov_list;
NodeVitastor *cli = NULL;
NodeVitastorImage *img = NULL; NodeVitastorImage *img = NULL;
int op = 0; int op = 0;
uint64_t offset = 0, len = 0, version = 0; uint64_t offset = 0, len = 0, version = 0;
@ -34,13 +32,6 @@ public:
Nan::Persistent<v8::Function> callback; Nan::Persistent<v8::Function> callback;
}; };
static uint64_t get_ui64(const v8::Local<v8::Value> & val)
{
if (val->IsBigInt())
return val->ToBigInt(Nan::GetCurrentContext()).ToLocalChecked()->Uint64Value();
return Nan::To<int64_t>(val).FromJust();
}
////////////////////////////////////////////////// //////////////////////////////////////////////////
// NodeVitastor // NodeVitastor
////////////////////////////////////////////////// //////////////////////////////////////////////////
@ -53,7 +44,6 @@ NodeVitastor::NodeVitastor(): Nan::ObjectWrap()
NodeVitastor::~NodeVitastor() NodeVitastor::~NodeVitastor()
{ {
TRACE("NodeVitastor: destructor");
uv_poll_stop(&poll_watcher); uv_poll_stop(&poll_watcher);
vitastor_c_destroy(c); vitastor_c_destroy(c);
c = NULL; c = NULL;
@ -85,10 +75,9 @@ NAN_METHOD(NodeVitastor::Create)
if (res >= 0) if (res >= 0)
{ {
cli->eventfd = res; cli->eventfd = res;
res = uv_poll_init(uv_default_loop(), &cli->poll_watcher, cli->eventfd); res = uv_poll_init_socket(uv_default_loop(), &cli->poll_watcher, cli->eventfd);
if (res >= 0) if (res >= 0)
res = uv_poll_start(&cli->poll_watcher, UV_READABLE, on_io_readable); res = uv_poll_start(&cli->poll_watcher, UV_READABLE, on_io_readable);
on_io_readable(&cli->poll_watcher, 0, UV_READABLE);
} }
if (res < 0) if (res < 0)
{ {
@ -109,14 +98,15 @@ void NodeVitastor::on_io_readable(uv_poll_t* handle, int status, int revents)
if (revents & UV_READABLE) if (revents & UV_READABLE)
{ {
NodeVitastor* self = (NodeVitastor*)handle->data; NodeVitastor* self = (NodeVitastor*)handle->data;
std::unique_lock<std::mutex> lock(self->mu);
vitastor_c_uring_handle_events(self->c); vitastor_c_uring_handle_events(self->c);
} }
} }
NodeVitastorRequest* NodeVitastor::get_read_request(const Nan::FunctionCallbackInfo<v8::Value> & info, int argpos) static NodeVitastorRequest* getReadRequest(const Nan::FunctionCallbackInfo<v8::Value> & info, int argpos)
{ {
uint64_t offset = get_ui64(info[argpos+0]); uint64_t offset = Nan::To<int64_t>(info[argpos+0]).FromJust();
uint64_t len = get_ui64(info[argpos+1]); uint64_t len = Nan::To<int64_t>(info[argpos+1]).FromJust();
uint8_t *buf = (uint8_t*)malloc(len); uint8_t *buf = (uint8_t*)malloc(len);
if (!buf) if (!buf)
{ {
@ -124,7 +114,7 @@ NodeVitastorRequest* NodeVitastor::get_read_request(const Nan::FunctionCallbackI
return NULL; return NULL;
} }
v8::Local<v8::Function> callback = info[argpos+2].As<v8::Function>(); v8::Local<v8::Function> callback = info[argpos+2].As<v8::Function>();
auto req = new NodeVitastorRequest(this, callback); auto req = new NodeVitastorRequest(callback);
req->offset = offset; req->offset = offset;
req->len = len; req->len = len;
@ -137,39 +127,35 @@ NodeVitastorRequest* NodeVitastor::get_read_request(const Nan::FunctionCallbackI
NAN_METHOD(NodeVitastor::Read) NAN_METHOD(NodeVitastor::Read)
{ {
TRACE("NodeVitastor::Read"); TRACE("NodeVitastor::Read");
if (info.Length() < 5)
Nan::ThrowError("Not enough arguments to read(pool, inode, offset, len, callback(err, buffer, version))");
NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This()); NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This());
uint64_t pool = get_ui64(info[0]); uint64_t pool = Nan::To<int64_t>(info[0]).FromJust();
uint64_t inode = get_ui64(info[1]); uint64_t inode = Nan::To<int64_t>(info[1]).FromJust();
auto req = self->get_read_request(info, 2); auto req = getReadRequest(info, 2);
self->Ref(); std::unique_lock<std::mutex> lock(self->mu);
vitastor_c_read(self->c, ((pool << (64-POOL_ID_BITS)) | inode), req->offset, req->len, &req->iov, 1, on_read_finish, req); vitastor_c_read(self->c, ((pool << (64-POOL_ID_BITS)) | inode), req->offset, req->len, &req->iov, 1, on_read_finish, req);
} }
NodeVitastorRequest* NodeVitastor::get_write_request(const Nan::FunctionCallbackInfo<v8::Value> & info, int argpos) static NodeVitastorRequest* getWriteRequest(const Nan::FunctionCallbackInfo<v8::Value> & info, int argpos)
{ {
uint64_t offset = get_ui64(info[argpos+0]); uint64_t offset = Nan::To<int64_t>(info[argpos+0]).FromJust();
const auto & bufarg = info[argpos+1]; const auto & bufarg = info[argpos+1];
uint64_t version = 0; uint64_t version = 0;
if (!info[argpos+2].IsEmpty() && if (!info[argpos+2].IsEmpty() && info[argpos+2]->IsObject())
!info[argpos+2]->IsFunction() &&
info[argpos+2]->IsObject())
{ {
auto key = Nan::New<v8::String>("version").ToLocalChecked(); auto key = Nan::New<v8::String>("version").ToLocalChecked();
auto params = info[argpos+2].As<v8::Object>(); auto params = info[argpos+2].As<v8::Object>();
auto versionObj = Nan::Get(params, key).ToLocalChecked(); auto versionObj = Nan::Get(params, key).ToLocalChecked();
if (!versionObj.IsEmpty()) if (!versionObj.IsEmpty())
version = get_ui64(versionObj); version = Nan::To<int64_t>(versionObj).FromJust();
argpos++; argpos++;
} }
v8::Local<v8::Function> callback = info[argpos+2].As<v8::Function>(); v8::Local<v8::Function> callback = info[argpos+2].As<v8::Function>();
auto req = new NodeVitastorRequest(this, callback); auto req = new NodeVitastorRequest(callback);
req->offset = offset; req->offset = offset;
req->version = version; req->version = version;
@ -191,7 +177,7 @@ NodeVitastorRequest* NodeVitastor::get_write_request(const Nan::FunctionCallback
{ {
char *buf = node::Buffer::Data(bufarg); char *buf = node::Buffer::Data(bufarg);
uint64_t len = node::Buffer::Length(bufarg); uint64_t len = node::Buffer::Length(bufarg);
req->iov = { .iov_base = buf, .iov_len = len }; req->iov = { .iov_base = buf, .iov_len = req->len };
req->len = len; req->len = len;
} }
@ -202,17 +188,15 @@ NodeVitastorRequest* NodeVitastor::get_write_request(const Nan::FunctionCallback
NAN_METHOD(NodeVitastor::Write) NAN_METHOD(NodeVitastor::Write)
{ {
TRACE("NodeVitastor::Write"); TRACE("NodeVitastor::Write");
if (info.Length() < 5)
Nan::ThrowError("Not enough arguments to write(pool, inode, offset, buf: Buffer | Buffer[], { version }?, callback(err))");
NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This()); NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This());
uint64_t pool = get_ui64(info[0]); uint64_t pool = Nan::To<int64_t>(info[0]).FromJust();
uint64_t inode = get_ui64(info[1]); uint64_t inode = Nan::To<int64_t>(info[1]).FromJust();
auto req = self->get_write_request(info, 2); auto req = getWriteRequest(info, 2);
self->Ref(); std::unique_lock<std::mutex> lock(self->mu);
vitastor_c_write(self->c, ((pool << (64-POOL_ID_BITS)) | inode), req->offset, req->len, req->version, vitastor_c_write(self->c, ((pool << (64-POOL_ID_BITS)) | inode), req->offset, req->len, req->version,
req->iov_list.size() ? req->iov_list.data() : &req->iov, req->iov_list.size() ? req->iov_list.data() : &req->iov,
req->iov_list.size() ? req->iov_list.size() : 1, req->iov_list.size() ? req->iov_list.size() : 1,
@ -223,15 +207,13 @@ NAN_METHOD(NodeVitastor::Write)
NAN_METHOD(NodeVitastor::Sync) NAN_METHOD(NodeVitastor::Sync)
{ {
TRACE("NodeVitastor::Sync"); TRACE("NodeVitastor::Sync");
if (info.Length() < 1)
Nan::ThrowError("Not enough arguments to sync(callback(err))");
NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This()); NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This());
v8::Local<v8::Function> callback = info[0].As<v8::Function>(); v8::Local<v8::Function> callback = info[0].As<v8::Function>();
auto req = new NodeVitastorRequest(self, callback); auto req = new NodeVitastorRequest(callback);
self->Ref(); std::unique_lock<std::mutex> lock(self->mu);
vitastor_c_sync(self->c, on_write_finish, req); vitastor_c_sync(self->c, on_write_finish, req);
} }
@ -239,20 +221,17 @@ NAN_METHOD(NodeVitastor::Sync)
NAN_METHOD(NodeVitastor::ReadBitmap) NAN_METHOD(NodeVitastor::ReadBitmap)
{ {
TRACE("NodeVitastor::ReadBitmap"); TRACE("NodeVitastor::ReadBitmap");
if (info.Length() < 6)
Nan::ThrowError("Not enough arguments to read_bitmap(pool, inode, offset, len, with_parents, callback(err, bitmap_buffer))");
NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This()); NodeVitastor* self = Nan::ObjectWrap::Unwrap<NodeVitastor>(info.This());
uint64_t pool = get_ui64(info[0]); uint64_t pool = Nan::To<int64_t>(info[0]).FromJust();
uint64_t inode = get_ui64(info[1]); uint64_t inode = Nan::To<int64_t>(info[1]).FromJust();
uint64_t offset = get_ui64(info[2]); uint64_t offset = Nan::To<int64_t>(info[2]).FromJust();
uint64_t len = get_ui64(info[3]); uint64_t len = Nan::To<int64_t>(info[3]).FromJust();
bool with_parents = Nan::To<bool>(info[4]).FromJust(); bool with_parents = Nan::To<bool>(info[4]).FromJust();
v8::Local<v8::Function> callback = info[5].As<v8::Function>(); v8::Local<v8::Function> callback = info[5].As<v8::Function>();
auto req = new NodeVitastorRequest(self, callback); auto req = new NodeVitastorRequest(callback);
self->Ref();
vitastor_c_read_bitmap(self->c, ((pool << (64-POOL_ID_BITS)) | inode), offset, len, with_parents, on_read_bitmap_finish, req); vitastor_c_read_bitmap(self->c, ((pool << (64-POOL_ID_BITS)) | inode), offset, len, with_parents, on_read_bitmap_finish, req);
} }
@ -269,7 +248,6 @@ static void on_error(NodeVitastorRequest *req, Nan::Callback & nanCallback, long
void NodeVitastor::on_read_finish(void *opaque, long retval, uint64_t version) void NodeVitastor::on_read_finish(void *opaque, long retval, uint64_t version)
{ {
TRACE("NodeVitastor::on_read_finish");
Nan::HandleScope scope; Nan::HandleScope scope;
NodeVitastorRequest *req = (NodeVitastorRequest *)opaque; NodeVitastorRequest *req = (NodeVitastorRequest *)opaque;
Nan::Callback nanCallback(Nan::New(req->callback)); Nan::Callback nanCallback(Nan::New(req->callback));
@ -278,7 +256,7 @@ void NodeVitastor::on_read_finish(void *opaque, long retval, uint64_t version)
free(req->iov.iov_base); free(req->iov.iov_base);
nanCallback.Call(0, NULL, req); nanCallback.Call(0, NULL, req);
} }
else if (retval < 0 || (uint64_t)retval != req->len) else if (retval < 0)
{ {
free(req->iov.iov_base); free(req->iov.iov_base);
on_error(req, nanCallback, retval); on_error(req, nanCallback, retval);
@ -291,33 +269,20 @@ void NodeVitastor::on_read_finish(void *opaque, long retval, uint64_t version)
args[2] = v8::BigInt::NewFromUnsigned(v8::Isolate::GetCurrent(), version); args[2] = v8::BigInt::NewFromUnsigned(v8::Isolate::GetCurrent(), version);
nanCallback.Call(3, args, req); nanCallback.Call(3, args, req);
} }
req->cli->Unref();
delete req; delete req;
} }
void NodeVitastor::on_write_finish(void *opaque, long retval) void NodeVitastor::on_write_finish(void *opaque, long retval)
{ {
TRACE("NodeVitastor::on_write_finish");
Nan::HandleScope scope; Nan::HandleScope scope;
NodeVitastorRequest *req = (NodeVitastorRequest *)opaque; NodeVitastorRequest *req = (NodeVitastorRequest *)opaque;
Nan::Callback nanCallback(Nan::New(req->callback)); Nan::Callback nanCallback(Nan::New(req->callback));
if (retval < 0 || (uint64_t)retval != req->len) on_error(req, nanCallback, retval);
{
on_error(req, nanCallback, retval);
}
else
{
v8::Local<v8::Value> args[1];
args[0] = Nan::Null();
nanCallback.Call(1, args, req);
}
req->cli->Unref();
delete req; delete req;
} }
void NodeVitastor::on_read_bitmap_finish(void *opaque, long retval, uint8_t *bitmap) void NodeVitastor::on_read_bitmap_finish(void *opaque, long retval, uint8_t *bitmap)
{ {
TRACE("NodeVitastor::on_read_bitmap_finish");
Nan::HandleScope scope; Nan::HandleScope scope;
NodeVitastorRequest *req = (NodeVitastorRequest *)opaque; NodeVitastorRequest *req = (NodeVitastorRequest *)opaque;
Nan::Callback nanCallback(Nan::New(req->callback)); Nan::Callback nanCallback(Nan::New(req->callback));
@ -332,7 +297,6 @@ void NodeVitastor::on_read_bitmap_finish(void *opaque, long retval, uint8_t *bit
args[1] = Nan::NewBuffer((char*)bitmap, (retval+7)/8).ToLocalChecked(); args[1] = Nan::NewBuffer((char*)bitmap, (retval+7)/8).ToLocalChecked();
nanCallback.Call(2, args, req); nanCallback.Call(2, args, req);
} }
req->cli->Unref();
delete req; delete req;
} }
@ -348,23 +312,21 @@ void NodeVitastor::on_read_bitmap_finish(void *opaque, long retval, uint8_t *bit
NAN_METHOD(NodeVitastorImage::Create) NAN_METHOD(NodeVitastorImage::Create)
{ {
TRACE("NodeVitastorImage::Create"); TRACE("NodeVitastorImage::Create");
if (info.Length() < 2)
Nan::ThrowError("Not enough arguments to Image(client, name)");
v8::Local<v8::Object> parent = info[0].As<v8::Object>(); v8::Local<v8::Object> parent = info[0].As<v8::Object>();
std::string name = std::string(*Nan::Utf8String(info[1].As<v8::String>())); std::string name = std::string(*Nan::Utf8String(info[1].As<v8::String>()));
NodeVitastor *cli = Nan::ObjectWrap::Unwrap<NodeVitastor>(parent); NodeVitastor *cli = Nan::ObjectWrap::Unwrap<NodeVitastor>(parent);
NodeVitastorImage *img = new NodeVitastorImage(); NodeVitastorImage *img = new NodeVitastorImage();
img->Wrap(info.This());
img->cli = cli; img->cli = cli;
img->name = name; img->name = name;
img->Ref(); img->Ref();
cli->Ref(); cli->Ref();
std::unique_lock<std::mutex> lock(cli->mu);
vitastor_c_watch_inode(cli->c, (char*)img->name.c_str(), on_watch_start, img); vitastor_c_watch_inode(cli->c, (char*)img->name.c_str(), on_watch_start, img);
img->Wrap(info.This());
info.GetReturnValue().Set(info.This()); info.GetReturnValue().Set(info.This());
} }
@ -382,12 +344,10 @@ NodeVitastorImage::~NodeVitastorImage()
NAN_METHOD(NodeVitastorImage::Read) NAN_METHOD(NodeVitastorImage::Read)
{ {
TRACE("NodeVitastorImage::Read"); TRACE("NodeVitastorImage::Read");
if (info.Length() < 3)
Nan::ThrowError("Not enough arguments to read(offset, len, callback(err, buffer, version))");
NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This()); NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This());
auto req = img->cli->get_read_request(info, 0); auto req = getReadRequest(info, 0);
req->img = img; req->img = img;
req->op = NODE_VITASTOR_READ; req->op = NODE_VITASTOR_READ;
@ -398,29 +358,24 @@ NAN_METHOD(NodeVitastorImage::Read)
NAN_METHOD(NodeVitastorImage::Write) NAN_METHOD(NodeVitastorImage::Write)
{ {
TRACE("NodeVitastorImage::Write"); TRACE("NodeVitastorImage::Write");
if (info.Length() < 3)
Nan::ThrowError("Not enough arguments to write(offset, buffer, { version }?, callback(err))");
NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This()); NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This());
auto req = img->cli->get_write_request(info, 0); auto req = getWriteRequest(info, 0);
req->img = img; req->img = img;
req->op = NODE_VITASTOR_WRITE; req->op = NODE_VITASTOR_WRITE;
img->exec_or_wait(req); img->exec_or_wait(req);
} }
// sync(callback(err))
NAN_METHOD(NodeVitastorImage::Sync) NAN_METHOD(NodeVitastorImage::Sync)
{ {
TRACE("NodeVitastorImage::Sync"); TRACE("NodeVitastorImage::Sync");
if (info.Length() < 1)
Nan::ThrowError("Not enough arguments to sync(callback(err))");
NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This()); NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This());
v8::Local<v8::Function> callback = info[0].As<v8::Function>(); v8::Local<v8::Function> callback = info[0].As<v8::Function>();
auto req = new NodeVitastorRequest(img->cli, callback); auto req = new NodeVitastorRequest(callback);
req->img = img; req->img = img;
req->op = NODE_VITASTOR_SYNC; req->op = NODE_VITASTOR_SYNC;
@ -431,17 +386,15 @@ NAN_METHOD(NodeVitastorImage::Sync)
NAN_METHOD(NodeVitastorImage::ReadBitmap) NAN_METHOD(NodeVitastorImage::ReadBitmap)
{ {
TRACE("NodeVitastorImage::ReadBitmap"); TRACE("NodeVitastorImage::ReadBitmap");
if (info.Length() < 4)
Nan::ThrowError("Not enough arguments to read_bitmap(offset, len, with_parents, callback(err, bitmap_buffer))");
NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This()); NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This());
uint64_t offset = get_ui64(info[0]); uint64_t offset = Nan::To<int64_t>(info[0]).FromJust();
uint64_t len = get_ui64(info[1]); uint64_t len = Nan::To<int64_t>(info[1]).FromJust();
bool with_parents = Nan::To<bool>(info[2]).FromJust(); bool with_parents = Nan::To<bool>(info[2]).FromJust();
v8::Local<v8::Function> callback = info[3].As<v8::Function>(); v8::Local<v8::Function> callback = info[3].As<v8::Function>();
auto req = new NodeVitastorRequest(img->cli, callback); auto req = new NodeVitastorRequest(callback);
req->img = img; req->img = img;
req->op = NODE_VITASTOR_READ_BITMAP; req->op = NODE_VITASTOR_READ_BITMAP;
req->offset = offset; req->offset = offset;
@ -451,17 +404,14 @@ NAN_METHOD(NodeVitastorImage::ReadBitmap)
img->exec_or_wait(req); img->exec_or_wait(req);
} }
// get_info(callback({ num, name, size, parent_id?, readonly?, meta?, mod_revision, block_size, bitmap_granularity, immediate_commit }))
NAN_METHOD(NodeVitastorImage::GetInfo) NAN_METHOD(NodeVitastorImage::GetInfo)
{ {
TRACE("NodeVitastorImage::GetInfo"); TRACE("NodeVitastorImage::Sync");
if (info.Length() < 1)
Nan::ThrowError("Not enough arguments to get_info(callback({ num, name, size, parent_id?, readonly?, meta?, mod_revision, block_size, bitmap_granularity, immediate_commit }))");
NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This()); NodeVitastorImage* img = Nan::ObjectWrap::Unwrap<NodeVitastorImage>(info.This());
v8::Local<v8::Function> callback = info[0].As<v8::Function>(); v8::Local<v8::Function> callback = info[0].As<v8::Function>();
auto req = new NodeVitastorRequest(img->cli, callback); auto req = new NodeVitastorRequest(callback);
req->img = img; req->img = img;
req->op = NODE_VITASTOR_GET_INFO; req->op = NODE_VITASTOR_GET_INFO;
@ -483,16 +433,15 @@ void NodeVitastorImage::exec_or_wait(NodeVitastorRequest *req)
void NodeVitastorImage::exec_request(NodeVitastorRequest *req) void NodeVitastorImage::exec_request(NodeVitastorRequest *req)
{ {
std::unique_lock<std::mutex> lock(cli->mu);
if (req->op == NODE_VITASTOR_READ) if (req->op == NODE_VITASTOR_READ)
{ {
uint64_t ino = vitastor_c_inode_get_num(watch); uint64_t ino = vitastor_c_inode_get_num(watch);
cli->Ref();
vitastor_c_read(cli->c, ino, req->offset, req->len, &req->iov, 1, NodeVitastor::on_read_finish, req); vitastor_c_read(cli->c, ino, req->offset, req->len, &req->iov, 1, NodeVitastor::on_read_finish, req);
} }
else if (req->op == NODE_VITASTOR_WRITE) else if (req->op == NODE_VITASTOR_WRITE)
{ {
uint64_t ino = vitastor_c_inode_get_num(watch); uint64_t ino = vitastor_c_inode_get_num(watch);
cli->Ref();
vitastor_c_write(cli->c, ino, req->offset, req->len, req->version, vitastor_c_write(cli->c, ino, req->offset, req->len, req->version,
req->iov_list.size() ? req->iov_list.data() : &req->iov, req->iov_list.size() ? req->iov_list.data() : &req->iov,
req->iov_list.size() ? req->iov_list.size() : 1, req->iov_list.size() ? req->iov_list.size() : 1,
@ -502,7 +451,6 @@ void NodeVitastorImage::exec_request(NodeVitastorRequest *req)
{ {
uint64_t ino = vitastor_c_inode_get_num(watch); uint64_t ino = vitastor_c_inode_get_num(watch);
uint32_t imm = vitastor_c_inode_get_immediate_commit(cli->c, ino); uint32_t imm = vitastor_c_inode_get_immediate_commit(cli->c, ino);
cli->Ref();
if (imm != IMMEDIATE_ALL) if (imm != IMMEDIATE_ALL)
{ {
vitastor_c_sync(cli->c, NodeVitastor::on_write_finish, req); vitastor_c_sync(cli->c, NodeVitastor::on_write_finish, req);
@ -515,7 +463,6 @@ void NodeVitastorImage::exec_request(NodeVitastorRequest *req)
else if (req->op == NODE_VITASTOR_READ_BITMAP) else if (req->op == NODE_VITASTOR_READ_BITMAP)
{ {
uint64_t ino = vitastor_c_inode_get_num(watch); uint64_t ino = vitastor_c_inode_get_num(watch);
cli->Ref();
vitastor_c_read_bitmap(cli->c, ino, req->offset, req->len, req->with_parents, NodeVitastor::on_read_bitmap_finish, req); vitastor_c_read_bitmap(cli->c, ino, req->offset, req->len, req->with_parents, NodeVitastor::on_read_bitmap_finish, req);
} }
else if (req->op == NODE_VITASTOR_GET_INFO) else if (req->op == NODE_VITASTOR_GET_INFO)
@ -585,14 +532,10 @@ void NodeVitastorImage::on_watch_start(void *opaque, long retval)
// NodeVitastorKV // NodeVitastorKV
////////////////////////////////////////////////// //////////////////////////////////////////////////
Nan::Persistent<v8::Function> NodeVitastorKV::listing_class;
// constructor(node_vitastor) // constructor(node_vitastor)
NAN_METHOD(NodeVitastorKV::Create) NAN_METHOD(NodeVitastorKV::Create)
{ {
TRACE("NodeVitastorKV::Create"); TRACE("NodeVitastorKV::Create");
if (info.Length() < 1)
Nan::ThrowError("Not enough arguments to new KV(client)");
v8::Local<v8::Object> parent = info[0].As<v8::Object>(); v8::Local<v8::Object> parent = info[0].As<v8::Object>();
NodeVitastor *cli = Nan::ObjectWrap::Unwrap<NodeVitastor>(parent); NodeVitastor *cli = Nan::ObjectWrap::Unwrap<NodeVitastor>(parent);
@ -600,32 +543,29 @@ NAN_METHOD(NodeVitastorKV::Create)
NodeVitastorKV *kv = new NodeVitastorKV(); NodeVitastorKV *kv = new NodeVitastorKV();
kv->cli = cli; kv->cli = cli;
{ {
std::unique_lock<std::mutex> lock(cli->mu);
kv->dbw = new vitastorkv_dbw_t((cluster_client_t*)vitastor_c_get_internal_client(cli->c)); kv->dbw = new vitastorkv_dbw_t((cluster_client_t*)vitastor_c_get_internal_client(cli->c));
} }
kv->Wrap(info.This()); kv->Wrap(info.This());
cli->Ref();
info.GetReturnValue().Set(info.This()); info.GetReturnValue().Set(info.This());
} }
NodeVitastorKV::~NodeVitastorKV() NodeVitastorKV::~NodeVitastorKV()
{ {
delete dbw; delete dbw;
cli->Unref();
} }
// open(pool_id, inode_num, { ...config }, callback(err)) // open(inode_id, { ...config }, callback(err))
NAN_METHOD(NodeVitastorKV::Open) NAN_METHOD(NodeVitastorKV::Open)
{ {
TRACE("NodeVitastorKV::Open"); TRACE("NodeVitastorKV::Open");
if (info.Length() < 4)
Nan::ThrowError("Not enough arguments to open(pool_id, inode_num, { ...config }, callback(err))");
NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This()); NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This());
uint64_t inode_id = INODE_WITH_POOL(get_ui64(info[0]), get_ui64(info[1])); uint64_t inode_id = Nan::To<int64_t>(info[0]).FromJust();
v8::Local<v8::Object> jsParams = info[2].As<v8::Object>(); v8::Local<v8::Object> jsParams = info[1].As<v8::Object>();
v8::Local<v8::Array> keys = Nan::GetOwnPropertyNames(jsParams).ToLocalChecked(); v8::Local<v8::Array> keys = Nan::GetOwnPropertyNames(jsParams).ToLocalChecked();
std::map<std::string, std::string> cfg; std::map<std::string, std::string> cfg;
for (uint32_t i = 0; i < keys->Length(); i++) for (uint32_t i = 0; i < keys->Length(); i++)
@ -634,8 +574,8 @@ NAN_METHOD(NodeVitastorKV::Open)
cfg[std::string(*Nan::Utf8String(key))] = std::string(*Nan::Utf8String(Nan::Get(jsParams, key).ToLocalChecked())); cfg[std::string(*Nan::Utf8String(key))] = std::string(*Nan::Utf8String(Nan::Get(jsParams, key).ToLocalChecked()));
} }
v8::Local<v8::Function> callback = info[3].As<v8::Function>(); v8::Local<v8::Function> callback = info[2].As<v8::Function>();
auto req = new NodeVitastorRequest(kv->cli, callback); auto req = new NodeVitastorRequest(callback);
kv->Ref(); kv->Ref();
kv->dbw->open(inode_id, cfg, [kv, req](int res) kv->dbw->open(inode_id, cfg, [kv, req](int res)
@ -654,13 +594,11 @@ NAN_METHOD(NodeVitastorKV::Open)
NAN_METHOD(NodeVitastorKV::Close) NAN_METHOD(NodeVitastorKV::Close)
{ {
TRACE("NodeVitastorKV::Close"); TRACE("NodeVitastorKV::Close");
if (info.Length() < 1)
Nan::ThrowError("Not enough arguments to close(callback(err))");
NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This()); NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This());
v8::Local<v8::Function> callback = info[0].As<v8::Function>(); v8::Local<v8::Function> callback = info[0].As<v8::Function>();
auto req = new NodeVitastorRequest(kv->cli, callback); auto req = new NodeVitastorRequest(callback);
kv->Ref(); kv->Ref();
kv->dbw->close([kv, req]() kv->dbw->close([kv, req]()
@ -677,8 +615,6 @@ NAN_METHOD(NodeVitastorKV::Close)
NAN_METHOD(NodeVitastorKV::SetConfig) NAN_METHOD(NodeVitastorKV::SetConfig)
{ {
TRACE("NodeVitastorKV::SetConfig"); TRACE("NodeVitastorKV::SetConfig");
if (info.Length() < 1)
Nan::ThrowError("Not enough arguments to set_config({ ...config })");
NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This()); NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This());
@ -715,7 +651,7 @@ void NodeVitastorKV::get_impl(const Nan::FunctionCallbackInfo<v8::Value> & info,
std::string key(*Nan::Utf8String(info[0].As<v8::String>())); std::string key(*Nan::Utf8String(info[0].As<v8::String>()));
v8::Local<v8::Function> callback = info[1].As<v8::Function>(); v8::Local<v8::Function> callback = info[1].As<v8::Function>();
auto req = new NodeVitastorRequest(kv->cli, callback); auto req = new NodeVitastorRequest(callback);
kv->Ref(); kv->Ref();
kv->dbw->get(key, [kv, req](int res, const std::string & value) kv->dbw->get(key, [kv, req](int res, const std::string & value)
@ -735,8 +671,6 @@ void NodeVitastorKV::get_impl(const Nan::FunctionCallbackInfo<v8::Value> & info,
NAN_METHOD(NodeVitastorKV::Get) NAN_METHOD(NodeVitastorKV::Get)
{ {
TRACE("NodeVitastorKV::Get"); TRACE("NodeVitastorKV::Get");
if (info.Length() < 2)
Nan::ThrowError("Not enough arguments to get(key, callback(err, value))");
get_impl(info, false); get_impl(info, false);
} }
@ -744,8 +678,6 @@ NAN_METHOD(NodeVitastorKV::Get)
NAN_METHOD(NodeVitastorKV::GetCached) NAN_METHOD(NodeVitastorKV::GetCached)
{ {
TRACE("NodeVitastorKV::GetCached"); TRACE("NodeVitastorKV::GetCached");
if (info.Length() < 2)
Nan::ThrowError("Not enough arguments to get_cached(key, callback(err, value))");
get_impl(info, true); get_impl(info, true);
} }
@ -764,12 +696,10 @@ static std::function<bool(int, const std::string &)> make_cas_callback(NodeVitas
}; };
} }
// set(key, value, callback(err), cas_compare(old_value)?) // set(key, value, callback(err), cas_compare(old_value))
NAN_METHOD(NodeVitastorKV::Set) NAN_METHOD(NodeVitastorKV::Set)
{ {
TRACE("NodeVitastorKV::Set"); TRACE("NodeVitastorKV::Set");
if (info.Length() < 3)
Nan::ThrowError("Not enough arguments to set(key, value, callback(err), cas_compare(old_value)?)");
NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This()); NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This());
@ -778,13 +708,13 @@ NAN_METHOD(NodeVitastorKV::Set)
std::string value(*Nan::Utf8String(info[1].As<v8::String>())); std::string value(*Nan::Utf8String(info[1].As<v8::String>()));
v8::Local<v8::Function> callback = info[2].As<v8::Function>(); v8::Local<v8::Function> callback = info[2].As<v8::Function>();
NodeVitastorRequest *req = new NodeVitastorRequest(kv->cli, callback), *cas_req = NULL; NodeVitastorRequest *req = new NodeVitastorRequest(callback), *cas_req = NULL;
std::function<bool(int, const std::string &)> cas_cb; std::function<bool(int, const std::string &)> cas_cb;
if (info.Length() > 3 && info[3]->IsObject()) if (info.Length() > 3 && info[3]->IsObject())
{ {
v8::Local<v8::Function> cas_callback = info[3].As<v8::Function>(); v8::Local<v8::Function> cas_callback = info[3].As<v8::Function>();
cas_req = new NodeVitastorRequest(kv->cli, cas_callback); cas_req = new NodeVitastorRequest(cas_callback);
cas_cb = make_cas_callback(cas_req); cas_cb = make_cas_callback(cas_req);
} }
@ -803,12 +733,10 @@ NAN_METHOD(NodeVitastorKV::Set)
}, cas_cb); }, cas_cb);
} }
// del(key, callback(err), cas_compare(old_value)?) // del(key, callback(err), cas_compare(old_value))
NAN_METHOD(NodeVitastorKV::Del) NAN_METHOD(NodeVitastorKV::Del)
{ {
TRACE("NodeVitastorKV::Del"); TRACE("NodeVitastorKV::Del");
if (info.Length() < 2)
Nan::ThrowError("Not enough arguments to del(key, callback(err), cas_compare(old_value)?)");
NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This()); NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This());
@ -816,13 +744,13 @@ NAN_METHOD(NodeVitastorKV::Del)
std::string key(*Nan::Utf8String(info[0].As<v8::String>())); std::string key(*Nan::Utf8String(info[0].As<v8::String>()));
v8::Local<v8::Function> callback = info[1].As<v8::Function>(); v8::Local<v8::Function> callback = info[1].As<v8::Function>();
NodeVitastorRequest *req = new NodeVitastorRequest(kv->cli, callback), *cas_req = NULL; NodeVitastorRequest *req = new NodeVitastorRequest(callback), *cas_req = NULL;
std::function<bool(int, const std::string &)> cas_cb; std::function<bool(int, const std::string &)> cas_cb;
if (info.Length() > 2 && info[2]->IsObject()) if (info.Length() > 2 && info[2]->IsObject())
{ {
v8::Local<v8::Function> cas_callback = info[2].As<v8::Function>(); v8::Local<v8::Function> cas_callback = info[2].As<v8::Function>();
cas_req = new NodeVitastorRequest(kv->cli, cas_callback); cas_req = new NodeVitastorRequest(cas_callback);
cas_cb = make_cas_callback(cas_req); cas_cb = make_cas_callback(cas_req);
} }
@ -858,14 +786,6 @@ NAN_METHOD(NodeVitastorKV::List)
info.GetReturnValue().Set(Nan::NewInstance(cons, narg, args).ToLocalChecked()); info.GetReturnValue().Set(Nan::NewInstance(cons, narg, args).ToLocalChecked());
} }
/*NAN_METHOD(NodeVitastorKV::Destroy)
{
TRACE("NodeVitastorKV::Destroy");
NodeVitastorKV* kv = Nan::ObjectWrap::Unwrap<NodeVitastorKV>(info.This());
if (!kv->dead)
kv->Unref();
}*/
////////////////////////////////////////////////// //////////////////////////////////////////////////
// NodeVitastorKVListing // NodeVitastorKVListing
////////////////////////////////////////////////// //////////////////////////////////////////////////
@ -887,10 +807,12 @@ NAN_METHOD(NodeVitastorKVListing::Create)
NodeVitastorKVListing *list = new NodeVitastorKVListing(); NodeVitastorKVListing *list = new NodeVitastorKVListing();
list->kv = kv; list->kv = kv;
list->handle = list->kv->dbw->list_start(start_key); {
std::unique_lock<std::mutex> lock(kv->cli->mu);
list->handle = list->kv->dbw->list_start(start_key);
}
list->Wrap(info.This()); list->Wrap(info.This());
kv->Ref();
info.GetReturnValue().Set(info.This()); info.GetReturnValue().Set(info.This());
} }
@ -898,54 +820,35 @@ NodeVitastorKVListing::~NodeVitastorKVListing()
{ {
if (handle) if (handle)
{ {
std::unique_lock<std::mutex> lock(kv->cli->mu);
kv->dbw->list_close(handle); kv->dbw->list_close(handle);
handle = NULL; handle = NULL;
} }
if (iter)
{
delete iter;
iter = NULL;
}
kv->Unref();
} }
// next(callback(err, value)?) // next(callback(err, value))
NAN_METHOD(NodeVitastorKVListing::Next) NAN_METHOD(NodeVitastorKVListing::Next)
{ {
TRACE("NodeVitastorKVListing::Next"); TRACE("NodeVitastorKVListing::Next");
NodeVitastorKVListing* list = Nan::ObjectWrap::Unwrap<NodeVitastorKVListing>(info.This()); NodeVitastorKVListing* list = Nan::ObjectWrap::Unwrap<NodeVitastorKVListing>(info.This());
if (info.Length() > 0) v8::Local<v8::Function> callback = info[0].As<v8::Function>();
{ auto req = new NodeVitastorRequest(callback);
v8::Local<v8::Function> callback = info[0].As<v8::Function>();
if (list->iter)
{
delete list->iter;
}
list->iter = new NodeVitastorRequest(list->kv->cli, callback);
}
if (!list->handle) if (!list->handle)
{ {
// Already closed // Already closed
if (list->iter) Nan::Callback nanCallback(Nan::New(req->callback));
{ v8::Local<v8::Value> args[1];
auto req = list->iter; args[0] = Nan::New<v8::Int32>(-EINVAL);
list->iter = NULL; nanCallback.Call(1, args, req);
Nan::Callback nanCallback(Nan::New(req->callback)); delete req;
v8::Local<v8::Value> args[1];
args[0] = Nan::New<v8::Int32>(-EINVAL);
nanCallback.Call(1, args, req);
delete req;
}
return; return;
} }
list->kv->Ref(); list->kv->Ref();
list->kv->dbw->list_next(list->handle, [list](int res, const std::string & key, const std::string & value) list->kv->dbw->list_next(list->handle, [list, req](int res, const std::string & key, const std::string & value)
{ {
auto req = list->iter;
list->iter = NULL;
Nan::HandleScope scope; Nan::HandleScope scope;
Nan::Callback nanCallback(Nan::New(req->callback)); Nan::Callback nanCallback(Nan::New(req->callback));
v8::Local<v8::Value> args[3]; v8::Local<v8::Value> args[3];
@ -953,10 +856,7 @@ NAN_METHOD(NodeVitastorKVListing::Next)
args[1] = !res ? v8::Local<v8::Value>(Nan::New<v8::String>(key).ToLocalChecked()) : v8::Local<v8::Value>(Nan::Null()); args[1] = !res ? v8::Local<v8::Value>(Nan::New<v8::String>(key).ToLocalChecked()) : v8::Local<v8::Value>(Nan::Null());
args[2] = !res ? v8::Local<v8::Value>(Nan::New<v8::String>(value).ToLocalChecked()) : v8::Local<v8::Value>(Nan::Null()); args[2] = !res ? v8::Local<v8::Value>(Nan::New<v8::String>(value).ToLocalChecked()) : v8::Local<v8::Value>(Nan::Null());
nanCallback.Call(3, args, req); nanCallback.Call(3, args, req);
if (list->iter) delete req;
delete req;
else
list->iter = req;
list->kv->Unref(); list->kv->Unref();
}); });
} }
@ -970,12 +870,8 @@ NAN_METHOD(NodeVitastorKVListing::Close)
if (list->handle) if (list->handle)
{ {
std::unique_lock<std::mutex> lock(list->kv->cli->mu);
list->kv->dbw->list_close(list->handle); list->kv->dbw->list_close(list->handle);
list->handle = NULL; list->handle = NULL;
} }
if (list->iter)
{
delete list->iter;
list->iter = NULL;
}
} }

View File

@ -4,6 +4,8 @@
#ifndef NODE_VITASTOR_CLIENT_H #ifndef NODE_VITASTOR_CLIENT_H
#define NODE_VITASTOR_CLIENT_H #define NODE_VITASTOR_CLIENT_H
#include <mutex>
#include <nan.h> #include <nan.h>
#include <vitastor_c.h> #include <vitastor_c.h>
#include <vitastor_kv.h> #include <vitastor_kv.h>
@ -32,6 +34,8 @@ private:
vitastor_c *c = NULL; vitastor_c *c = NULL;
int eventfd = -1; int eventfd = -1;
uv_poll_t poll_watcher; uv_poll_t poll_watcher;
// FIXME: Is it really needed?
std::mutex mu;
NodeVitastor(); NodeVitastor();
@ -40,9 +44,6 @@ private:
static void on_write_finish(void *opaque, long retval); static void on_write_finish(void *opaque, long retval);
static void on_read_bitmap_finish(void *opaque, long retval, uint8_t *bitmap); static void on_read_bitmap_finish(void *opaque, long retval, uint8_t *bitmap);
NodeVitastorRequest* get_read_request(const Nan::FunctionCallbackInfo<v8::Value> & info, int argpos);
NodeVitastorRequest* get_write_request(const Nan::FunctionCallbackInfo<v8::Value> & info, int argpos);
friend class NodeVitastorImage; friend class NodeVitastorImage;
friend class NodeVitastorKV; friend class NodeVitastorKV;
friend class NodeVitastorKVListing; friend class NodeVitastorKVListing;
@ -73,6 +74,8 @@ private:
std::vector<NodeVitastorRequest*> on_init; std::vector<NodeVitastorRequest*> on_init;
Nan::Persistent<v8::Object> cliObj; Nan::Persistent<v8::Object> cliObj;
NodeVitastorImage();
static void on_watch_start(void *opaque, long retval); static void on_watch_start(void *opaque, long retval);
void exec_request(NodeVitastorRequest *req); void exec_request(NodeVitastorRequest *req);
void exec_or_wait(NodeVitastorRequest *req); void exec_or_wait(NodeVitastorRequest *req);
@ -83,7 +86,7 @@ class NodeVitastorKV: public Nan::ObjectWrap
public: public:
// constructor(node_vitastor) // constructor(node_vitastor)
static NAN_METHOD(Create); static NAN_METHOD(Create);
// open(pool_id, inode_num, { ...config }, callback(err)) // open(inode_id, { ...config }, callback(err))
static NAN_METHOD(Open); static NAN_METHOD(Open);
// set_config({ ...config }) // set_config({ ...config })
static NAN_METHOD(SetConfig); static NAN_METHOD(SetConfig);
@ -95,9 +98,9 @@ public:
static NAN_METHOD(Get); static NAN_METHOD(Get);
// get_cached(key, callback(err, value)) // get_cached(key, callback(err, value))
static NAN_METHOD(GetCached); static NAN_METHOD(GetCached);
// set(key, value, callback(err), cas_compare(old_value)?) // set(key, value, callback(err), cas_compare(old_value))
static NAN_METHOD(Set); static NAN_METHOD(Set);
// del(key, callback(err), cas_compare(old_value)?) // del(key, callback(err), cas_compare(old_value))
static NAN_METHOD(Del); static NAN_METHOD(Del);
// list(start_key?) // list(start_key?)
static NAN_METHOD(List); static NAN_METHOD(List);
@ -110,6 +113,8 @@ private:
NodeVitastor *cli = NULL; NodeVitastor *cli = NULL;
vitastorkv_dbw_t *dbw = NULL; vitastorkv_dbw_t *dbw = NULL;
NodeVitastorKV();
static void get_impl(const Nan::FunctionCallbackInfo<v8::Value> & info, bool allow_cache); static void get_impl(const Nan::FunctionCallbackInfo<v8::Value> & info, bool allow_cache);
friend class NodeVitastorKVListing; friend class NodeVitastorKVListing;
@ -120,7 +125,7 @@ class NodeVitastorKVListing: public Nan::ObjectWrap
public: public:
// constructor(node_vitastor_kv, start_key?) // constructor(node_vitastor_kv, start_key?)
static NAN_METHOD(Create); static NAN_METHOD(Create);
// next(callback(err, value)?) // next(callback(err, value))
static NAN_METHOD(Next); static NAN_METHOD(Next);
// close() // close()
static NAN_METHOD(Close); static NAN_METHOD(Close);
@ -130,7 +135,8 @@ public:
private: private:
NodeVitastorKV *kv = NULL; NodeVitastorKV *kv = NULL;
void *handle = NULL; void *handle = NULL;
NodeVitastorRequest *iter = NULL;
NodeVitastorKVListing();
}; };
#endif #endif

View File

@ -1 +0,0 @@
module.exports = require('bindings')('addon.node');

View File

@ -1286,11 +1286,7 @@ void cluster_client_t::handle_op_part(cluster_op_part_t *part)
if (op->opcode == OSD_OP_READ || op->opcode == OSD_OP_READ_BITMAP || op->opcode == OSD_OP_READ_CHAIN_BITMAP) if (op->opcode == OSD_OP_READ || op->opcode == OSD_OP_READ_BITMAP || op->opcode == OSD_OP_READ_CHAIN_BITMAP)
{ {
copy_part_bitmap(op, part); copy_part_bitmap(op, part);
if (op->inode == op->cur_inode) op->version = op->parts.size() == 1 ? part->op.reply.rw.version : 0;
{
// Read only returns the version of the uppermost layer
op->version = op->parts.size() == 1 ? part->op.reply.rw.version : 0;
}
} }
else if (op->opcode == OSD_OP_WRITE) else if (op->opcode == OSD_OP_WRITE)
{ {