1
0
Fork 0
mirror of https://github.com/rubyjs/therubyracer synced 2023-03-27 23:21:42 -04:00
therubyracer/ext/v8/v8_cxt.cpp

137 lines
3.7 KiB
C++

#include "rr.h"
#include "v8_cxt.h"
#include "v8_msg.h"
#include "v8_template.h"
#include "converters.h"
using namespace v8;
VALUE V8_C_Context;
//TODO: rename everything to Context_
//TODO: do the object init from within here
namespace {
Local<Context> unwrap(VALUE value) {
return V8_Ref_Get<Context>(value);
}
VALUE New(int argc, VALUE *argv, VALUE self) {
HandleScope handles;
VALUE scope;
rb_scan_args(argc,argv, "01", &scope);
if (NIL_P(scope)) {
return V8_Ref_Create(self, Context::New());
} else {
Persistent<Context> context = Context::New(0, Racer_Create_V8_ObjectTemplate(scope));
Context::Scope enter(context);
context->Global()->SetHiddenValue(String::New("TheRubyRacer::RubyObject"), External::Wrap((void *)scope));
VALUE ref = V8_Ref_Create(self, context, scope);
context.Dispose();
return ref;
}
}
VALUE InContext(VALUE self) {
return Context::InContext() ? Qtrue : Qfalse;
}
VALUE GetEntered(VALUE self) {
HandleScope handles;
if (Context::InContext()) {
Local<Context> current = Context::GetEntered();
return V8_Ref_Create(self, current);
} else {
return Qnil;
}
}
VALUE Global(VALUE self) {
HandleScope handles;
return rr_v82rb(unwrap(self)->Global());
}
VALUE Enter(VALUE self) {
unwrap(self)->Enter();
return self;
}
VALUE Exit(VALUE self) {
unwrap(self)->Exit();
return self;
}
VALUE v8_cxt_open(VALUE self) {
HandleScope handles;
Local<Context> cxt = V8_Ref_Get<Context>(self);
Context::Scope enter(cxt);
if (rb_block_given_p()) {
return rb_yield(self);
} else {
return Qnil;
}
}
VALUE Racer_Error_Message(TryCatch& exception) {
VALUE msg = V8_Wrap_Message(exception.Message());
Local<Value> stack = exception.StackTrace();
if (!stack.IsEmpty()) {
rb_iv_set(msg,"@stack",V82RB(stack));
}
return msg;
}
VALUE v8_cxt_eval(VALUE self, VALUE source, VALUE filename) {
HandleScope handles;
TryCatch exceptions;
Local<Context> cxt = V8_Ref_Get<Context>(self);
Context::Scope enter(cxt);
Local<Value> source_str = RB2V8(source);
Local<Value> source_name = RTEST(filename) ? RB2V8(filename) : *String::New("<eval>");
Local<Script> script = Script::Compile(source_str->ToString(), source_name);
if (exceptions.HasCaught()) {
return Racer_Error_Message(exceptions);
}
Local<Value> result = script->Run();
if (exceptions.HasCaught()) {
return Racer_Error_Message(exceptions);
} else {
return rr_v82rb(result);
}
}
VALUE v8_cxt_eql(VALUE self, VALUE other) {
HandleScope handles;
if (RTEST(CLASS_OF(other) != V8_C_Context)) {
return Qnil;
} else {
Local<Context> cxt = V8_Ref_Get<Context>(self);
Local<Context> that = V8_Ref_Get<Context>(other);
return cxt == that ? Qtrue : Qfalse;
}
return Qnil;
}
}
void rr_init_cxt() {
VALUE Context = V8_C_Context = rr_define_class("Context");
rr_define_singleton_method(Context, "new", New, -1);
rr_define_singleton_method(Context, "New", New, -1);
rr_define_singleton_method(Context, "InContext", InContext, 0);
rr_define_singleton_method(Context, "GetEntered", GetEntered, 0);
rr_define_method(Context, "Global", Global, 0);
rr_define_method(Context, "Enter", Enter, 0);
rr_define_method(Context, "Exit", Exit, 0);
rr_define_method(Context, "open", v8_cxt_open, 0);
rr_define_method(Context, "eval", v8_cxt_eval, 2);
rr_define_method(Context, "eql?", v8_cxt_eql, 1);
rr_define_method(Context, "==", v8_cxt_eql, 1);
}
VALUE rr_reflect_v8_context(Handle<Context> value) {
return V8_Ref_Create(V8_C_Context, value);
}