#include "events.h" #include #include #include #include #include #include #include #include #include #include #include /* inet_ntop */ #include /* sockaddr_in, sockaddr_in6 */ #ifndef RAMP # define RAMP(x) ((x) > 0 ? (x) : 0) #endif using namespace v8; using namespace node; Persistent EventEmitter::constructor_template; void EventEmitter::Initialize (Local ctemplate) { HandleScope scope; constructor_template = Persistent::New(ctemplate); Local __emit = FunctionTemplate::New(Emit); constructor_template->PrototypeTemplate()->Set(String::NewSymbol("emit"), __emit); // All other prototype methods are defined in events.js } static bool ReallyEmit (Handle self, Handle event, int argc, Handle argv[]) { HandleScope scope; Local events_v = self->Get(String::NewSymbol("_events")); if (!events_v->IsObject()) return false; Local events = events_v->ToObject(); Local listeners_v = events->Get(event); if (!listeners_v->IsArray()) return false; Local listeners = Local::Cast(listeners_v); for (unsigned int i = 0; i < listeners->Length(); i++) { HandleScope scope; Local listener_v = listeners->Get(Integer::New(i)); if (!listener_v->IsFunction()) continue; Local listener = Local::Cast(listener_v); TryCatch try_catch; listener->Call(self, argc, argv); if (try_catch.HasCaught()) { FatalException(try_catch); return false; } } return true; } Handle EventEmitter::Emit (const Arguments& args) { HandleScope scope; Local event = args[0]->ToString(); int argc = 0; Local emit_args; if (args[1]->IsArray()) { emit_args = Local::Cast(args[1]); argc = emit_args->Length(); } Local argv[argc]; for (int i = 0; i < argc; i++) { argv[i] = emit_args->Get(Integer::New(i)); } bool r = ReallyEmit(args.Holder(), event, argc, argv); return scope.Close(r ? True() : False()); } bool EventEmitter::Emit (const char *event_s, int argc, Handle argv[]) { HandleScope scope; Local event = String::NewSymbol(event_s); return ReallyEmit(handle_, event, argc, argv); } Persistent Promise::constructor_template; void Promise::Initialize (v8::Handle target) { HandleScope scope; Local t = FunctionTemplate::New(New); constructor_template = Persistent::New(t); constructor_template->Inherit(EventEmitter::constructor_template); constructor_template->InstanceTemplate()->SetInternalFieldCount(1); NODE_SET_PROTOTYPE_METHOD(constructor_template, "block", Block); NODE_SET_PROTOTYPE_METHOD(constructor_template, "emitSuccess", EmitSuccess); NODE_SET_PROTOTYPE_METHOD(constructor_template, "emitError", EmitError); target->Set(String::NewSymbol("Promise"), constructor_template->GetFunction()); } v8::Handle Promise::New (const v8::Arguments& args) { HandleScope scope; Promise *promise = new Promise(); promise->Wrap(args.This()); promise->Attach(); return args.This(); } Handle Promise::Block (const Arguments& args) { HandleScope scope; Promise *promise = ObjectWrap::Unwrap(args.Holder()); promise->Block(); return Undefined(); } v8::Handle Promise::EmitSuccess (const v8::Arguments& args) { HandleScope scope; Promise *promise = ObjectWrap::Unwrap(args.Holder()); int argc = 0; Local emit_args; if (args[0]->IsArray()) { emit_args = Local::Cast(args[0]); argc = emit_args->Length(); } Local argv[argc]; for (int i = 0; i < argc; i++) { argv[i] = emit_args->Get(Integer::New(i)); } bool r = promise->EmitSuccess(argc, argv); return r ? True() : False(); } v8::Handle Promise::EmitError (const v8::Arguments& args) { HandleScope scope; Promise *promise = ObjectWrap::Unwrap(args.Holder()); int argc = 0; Local emit_args; if (args[0]->IsArray()) { emit_args = Local::Cast(args[0]); argc = emit_args->Length(); } Local argv[argc]; for (int i = 0; i < argc; i++) { argv[i] = emit_args->Get(Integer::New(i)); } bool r = promise->EmitError(argc, argv); return r ? True() : False(); } void Promise::Block (void) { blocking_ = true; ev_loop(EV_DEFAULT_UC_ 0); } void Promise::Detach (void) { if (blocking_) { ev_unloop(EV_DEFAULT_ EVUNLOOP_ONE); blocking_ = false; } if (ref_) { ev_unref(EV_DEFAULT_UC); } ObjectWrap::Detach(); } Promise* Promise::Create (bool ref) { HandleScope scope; Local handle = Promise::constructor_template->GetFunction()->NewInstance(); Promise *promise = ObjectWrap::Unwrap(handle); promise->ref_ = ref; if (ref) ev_ref(EV_DEFAULT_UC); return promise; } bool Promise::EmitSuccess (int argc, v8::Handle argv[]) { bool r = Emit("success", argc, argv); Detach(); return r; } bool Promise::EmitError (int argc, v8::Handle argv[]) { bool r = Emit("error", argc, argv); Detach(); return r; }