From 470c6342ccb3c74c96e5764f0aaa9ebf6e1d2d94 Mon Sep 17 00:00:00 2001 From: Ryan Date: Thu, 16 Apr 2009 13:20:35 +0200 Subject: [PATCH] Change symantics of file.open. Now takes a callback argument. --- src/file.cc | 88 ++++++++++++++++++++++---------------------------- test/simple.js | 30 ++++++++--------- 2 files changed, 53 insertions(+), 65 deletions(-) diff --git a/src/file.cc b/src/file.cc index 4c881faed6..3b1f2123bf 100644 --- a/src/file.cc +++ b/src/file.cc @@ -9,6 +9,9 @@ using namespace v8; +#define ON_OPEN v8::String::NewSymbol("onOpen") + + class File; class Callback { public: @@ -156,9 +159,7 @@ public: static File* Unwrap (Handle obj); - void OnError (const char *syscall, int errorno); - - void Open (const char *path, const char *mode); + Handle Open (const char *path, const char *mode, Handle callback); static int AfterOpen (eio_req *req); void Close (); @@ -213,6 +214,11 @@ File::AfterClose (eio_req *req) { File *file = static_cast(req->data); + if (req->result == 0) { + file->fd_ = -1; + file->handle_->Delete(String::NewSymbol("fd")); + } + // TODO printf("after close\n"); return 0; @@ -225,77 +231,59 @@ File::Close () node_eio_submit(req); } -void -File::OnError (const char *syscall, int errorno) -{ - HandleScope scope; - - Handle on_error_value = handle_->Get( String::NewSymbol("onError") ); - if (!on_error_value->IsFunction()) - return; - Handle on_error = Handle::Cast(on_error_value); - - const int argc = 3; - Local argv[argc]; - - argv[0] = String::New(syscall); - argv[1] = Integer::New(errorno); - argv[2] = String::New(strerror(errorno)); - - TryCatch try_catch; - on_error->Call(handle_, argc, argv); - - if(try_catch.HasCaught()) - node_fatal_exception(try_catch); -} - int File::AfterOpen (eio_req *req) { File *file = static_cast(req->data); - HandleScope scope; - TryCatch try_catch; - if(req->result < 0) { - file->OnError("open", req->errorno); - return 0; + if(req->result >= 0) { + file->fd_ = static_cast(req->result); + file->handle_->Set(String::NewSymbol("fd"), Integer::New(file->fd_)); } - file->fd_ = static_cast(req->result); - - file->handle_->Set(String::NewSymbol("fd"), Integer::New(req->result)); - Handle on_open_value = file->handle_->Get( String::NewSymbol("onOpen") ); - if (!on_open_value->IsFunction()) + Handle callback_value = file->handle_->Get(ON_OPEN); + if (!callback_value->IsFunction()) return 0; - Handle on_open = Handle::Cast(on_open_value); + Handle callback = Handle::Cast(callback_value); + file->handle_->Delete(ON_OPEN); - on_open->Call(file->handle_, 0, NULL); + const int argc = 1; + Handle argv[argc]; + argv[0] = Integer::New(req->errorno); + TryCatch try_catch; + callback->Call(file->handle_, argc, argv); if(try_catch.HasCaught()) node_fatal_exception(try_catch); return 0; } -void -File::Open (const char *path, const char *mode) +Handle +File::Open (const char *path, const char *mode, Handle callback) { - mode_t mask = umask(0x0700); + // make sure that we don't already have a pending open + if (handle_->Has(ON_OPEN)) { + return ThrowException(String::New("File object is already being opened.")); + } + + if (callback->IsFunction()) handle_->Set(ON_OPEN, callback); + + // Get the current umask + mode_t mask = umask(0); umask(mask); int flags = O_RDONLY; // default - // XXX is this interpretation correct? - // I don't want to to use fopen() because eio doesn't support it. + // XXX is this interpretation of the mode correct? + // I don't want to to use fopen() directly because eio doesn't support it. switch(mode[0]) { case 'r': flags = (mode[1] == '+' ? O_RDWR : O_RDONLY); break; - case 'w': flags = O_CREAT | O_TRUNC | (mode[1] == '+' ? O_RDWR : O_WRONLY); break; - case 'a': flags = O_APPEND | O_CREAT | (mode[1] == '+' ? O_RDWR : O_WRONLY); break; @@ -303,6 +291,8 @@ File::Open (const char *path, const char *mode) eio_req *req = eio_open (path, flags, mask, EIO_PRI_DEFAULT, File::AfterOpen, this); node_eio_submit(req); + + return Undefined(); } int @@ -364,12 +354,10 @@ JS_METHOD(file_open) String::Utf8Value path(args[0]->ToString()); if (args[1]->IsString()) { String::AsciiValue mode(args[1]->ToString()); - file->Open(*path, *mode); + return file->Open(*path, *mode, args[2]); } else { - file->Open(*path, "r"); + return file->Open(*path, "r", args[1]); } - - return Undefined(); } JS_METHOD(file_close) diff --git a/test/simple.js b/test/simple.js index 36511888e2..6920f88bc7 100644 --- a/test/simple.js +++ b/test/simple.js @@ -1,18 +1,18 @@ var f = new File; -f.onError = function (call) { - node.blocking.print("file error with " + call ); -} -f.onOpen = function () { - node.blocking.print("file opened."); - f.write("hello world\n", function (status, written) { - node.blocking.print("written. status: " - + status.toString() - + " written: " - + written.toString() - ); - f.close(); - }); -}; +f.open("/tmp/world", "r+", function (status) { + if (status == 0) { + node.blocking.print("file opened"); + f.write("hello world\n", function (status, written) { -f.open("/tmp/world", "a+"); + node.blocking.print("written. status: " + + status.toString() + + " written: " + + written.toString() + ); + f.close(); + }); + } else { + node.blocking.print("file open failed: " + status.toString()); + } +});