jbarnette-johnson 1.0.0.200806240111
Sign up to get free protection for your applications and to get access to all the features.
- data/CHANGELOG +5 -0
- data/MANIFEST +385 -0
- data/MINGW32.mk +124 -0
- data/README.rdoc +51 -0
- data/Rakefile +166 -0
- data/bin/johnson +107 -0
- data/cross-compile.txt +38 -0
- data/ext/spidermonkey/context.c +122 -0
- data/ext/spidermonkey/context.h +19 -0
- data/ext/spidermonkey/conversions.c +286 -0
- data/ext/spidermonkey/conversions.h +18 -0
- data/ext/spidermonkey/debugger.c +208 -0
- data/ext/spidermonkey/debugger.h +9 -0
- data/ext/spidermonkey/extconf.rb +25 -0
- data/ext/spidermonkey/extensions.c +37 -0
- data/ext/spidermonkey/extensions.h +12 -0
- data/ext/spidermonkey/global.c +40 -0
- data/ext/spidermonkey/global.h +11 -0
- data/ext/spidermonkey/idhash.c +16 -0
- data/ext/spidermonkey/idhash.h +8 -0
- data/ext/spidermonkey/immutable_node.c.erb +522 -0
- data/ext/spidermonkey/immutable_node.h +22 -0
- data/ext/spidermonkey/jroot.h +187 -0
- data/ext/spidermonkey/js_land_proxy.c +609 -0
- data/ext/spidermonkey/js_land_proxy.h +20 -0
- data/ext/spidermonkey/ruby_land_proxy.c +537 -0
- data/ext/spidermonkey/ruby_land_proxy.h +17 -0
- data/ext/spidermonkey/runtime.c +304 -0
- data/ext/spidermonkey/runtime.h +25 -0
- data/ext/spidermonkey/spidermonkey.c +20 -0
- data/ext/spidermonkey/spidermonkey.h +29 -0
- data/js/johnson/browser.js +9 -0
- data/js/johnson/browser/env.js +687 -0
- data/js/johnson/browser/jquery.js +3444 -0
- data/js/johnson/browser/xmlsax.js +1564 -0
- data/js/johnson/browser/xmlw3cdom.js +4189 -0
- data/js/johnson/cli.js +30 -0
- data/js/johnson/prelude.js +80 -0
- data/js/johnson/template.js +29 -0
- data/lib/hoe.rb +748 -0
- data/lib/johnson.rb +46 -0
- data/lib/johnson/cli.rb +7 -0
- data/lib/johnson/cli/options.rb +56 -0
- data/lib/johnson/error.rb +4 -0
- data/lib/johnson/nodes.rb +7 -0
- data/lib/johnson/nodes/binary_node.rb +64 -0
- data/lib/johnson/nodes/for.rb +14 -0
- data/lib/johnson/nodes/for_in.rb +12 -0
- data/lib/johnson/nodes/function.rb +13 -0
- data/lib/johnson/nodes/list.rb +27 -0
- data/lib/johnson/nodes/node.rb +68 -0
- data/lib/johnson/nodes/ternary_node.rb +20 -0
- data/lib/johnson/parser.rb +21 -0
- data/lib/johnson/parser/syntax_error.rb +13 -0
- data/lib/johnson/runtime.rb +55 -0
- data/lib/johnson/spidermonkey/context.rb +10 -0
- data/lib/johnson/spidermonkey/debugger.rb +67 -0
- data/lib/johnson/spidermonkey/immutable_node.rb +280 -0
- data/lib/johnson/spidermonkey/js_land_proxy.rb +62 -0
- data/lib/johnson/spidermonkey/mutable_tree_visitor.rb +233 -0
- data/lib/johnson/spidermonkey/ruby_land_proxy.rb +52 -0
- data/lib/johnson/spidermonkey/runtime.rb +94 -0
- data/lib/johnson/version.rb +4 -0
- data/lib/johnson/visitable.rb +16 -0
- data/lib/johnson/visitors.rb +4 -0
- data/lib/johnson/visitors/dot_visitor.rb +167 -0
- data/lib/johnson/visitors/ecma_visitor.rb +315 -0
- data/lib/johnson/visitors/enumerating_visitor.rb +115 -0
- data/lib/johnson/visitors/sexp_visitor.rb +172 -0
- data/lib/rails/init.rb +37 -0
- data/test/assets/index.html +38 -0
- data/test/assets/jquery_test.html +186 -0
- data/test/helper.rb +58 -0
- data/test/johnson/browser_test.rb +38 -0
- data/test/johnson/conversions/array_test.rb +32 -0
- data/test/johnson/conversions/boolean_test.rb +17 -0
- data/test/johnson/conversions/callable_test.rb +34 -0
- data/test/johnson/conversions/file_test.rb +15 -0
- data/test/johnson/conversions/nil_test.rb +20 -0
- data/test/johnson/conversions/number_test.rb +34 -0
- data/test/johnson/conversions/regexp_test.rb +24 -0
- data/test/johnson/conversions/string_test.rb +26 -0
- data/test/johnson/conversions/struct_test.rb +15 -0
- data/test/johnson/conversions/symbol_test.rb +19 -0
- data/test/johnson/conversions/thread_test.rb +24 -0
- data/test/johnson/error_test.rb +9 -0
- data/test/johnson/extensions_test.rb +56 -0
- data/test/johnson/nodes/array_literal_test.rb +57 -0
- data/test/johnson/nodes/array_node_test.rb +26 -0
- data/test/johnson/nodes/binary_node_test.rb +61 -0
- data/test/johnson/nodes/bracket_access_test.rb +16 -0
- data/test/johnson/nodes/delete_test.rb +11 -0
- data/test/johnson/nodes/do_while_test.rb +12 -0
- data/test/johnson/nodes/dot_accessor_test.rb +15 -0
- data/test/johnson/nodes/export_test.rb +9 -0
- data/test/johnson/nodes/for_test.rb +54 -0
- data/test/johnson/nodes/function_test.rb +71 -0
- data/test/johnson/nodes/if_test.rb +41 -0
- data/test/johnson/nodes/import_test.rb +13 -0
- data/test/johnson/nodes/label_test.rb +19 -0
- data/test/johnson/nodes/object_literal_test.rb +110 -0
- data/test/johnson/nodes/return_test.rb +16 -0
- data/test/johnson/nodes/semi_test.rb +8 -0
- data/test/johnson/nodes/switch_test.rb +55 -0
- data/test/johnson/nodes/ternary_test.rb +25 -0
- data/test/johnson/nodes/throw_test.rb +9 -0
- data/test/johnson/nodes/try_node_test.rb +59 -0
- data/test/johnson/nodes/typeof_test.rb +11 -0
- data/test/johnson/nodes/unary_node_test.rb +23 -0
- data/test/johnson/nodes/void_test.rb +11 -0
- data/test/johnson/nodes/while_test.rb +26 -0
- data/test/johnson/nodes/with_test.rb +10 -0
- data/test/johnson/prelude_test.rb +56 -0
- data/test/johnson/runtime_test.rb +46 -0
- data/test/johnson/spidermonkey/context_test.rb +21 -0
- data/test/johnson/spidermonkey/immutable_node_test.rb +34 -0
- data/test/johnson/spidermonkey/js_land_proxy_test.rb +236 -0
- data/test/johnson/spidermonkey/ruby_land_proxy_test.rb +225 -0
- data/test/johnson/spidermonkey/runtime_test.rb +17 -0
- data/test/johnson/version_test.rb +13 -0
- data/test/johnson/visitors/dot_visitor_test.rb +39 -0
- data/test/johnson/visitors/enumerating_visitor_test.rb +12 -0
- data/test/johnson_test.rb +16 -0
- data/test/jquery_units/test.js +27 -0
- data/test/jquery_units/test_helper.js +197 -0
- data/test/jquery_units/units/ajax.js +795 -0
- data/test/jquery_units/units/core.js +1563 -0
- data/test/jquery_units/units/event.js +299 -0
- data/test/jquery_units/units/fx.js +427 -0
- data/test/jquery_units/units/offset.js +112 -0
- data/test/jquery_units/units/selector.js +224 -0
- data/test/jspec/helper.js +7 -0
- data/test/jspec/jspec.js +192 -0
- data/test/jspec/simple_spec.js +68 -0
- data/test/parser_test.rb +276 -0
- data/todo/.keep +0 -0
- metadata +501 -0
@@ -0,0 +1,22 @@
|
|
1
|
+
#ifndef JOHNSON_SPIDERMONKEY_IMMUTABLE_NODE_H
|
2
|
+
#define JOHNSON_SPIDERMONKEY_IMMUTABLE_NODE_H
|
3
|
+
|
4
|
+
#include "spidermonkey.h"
|
5
|
+
#include "jsparse.h"
|
6
|
+
#include "jsatom.h"
|
7
|
+
#include "jsscan.h"
|
8
|
+
#include "jsarena.h"
|
9
|
+
#include "jsfun.h"
|
10
|
+
#include "jscntxt.h"
|
11
|
+
|
12
|
+
typedef struct {
|
13
|
+
JSParseContext * pc;
|
14
|
+
JSParseNode * node;
|
15
|
+
JSContext * js;
|
16
|
+
JSRuntime * runtime;
|
17
|
+
} ImmutableNodeContext;
|
18
|
+
|
19
|
+
VALUE jsop_to_symbol(JSUint32 jsop);
|
20
|
+
void init_Johnson_SpiderMonkey_Immutable_Node(VALUE spidermonkey);
|
21
|
+
|
22
|
+
#endif
|
@@ -0,0 +1,187 @@
|
|
1
|
+
#ifndef JOHNSON_JROOT_H
|
2
|
+
#define JOHNSON_JROOT_H
|
3
|
+
|
4
|
+
#define _JROOT_NAMESIZE 200
|
5
|
+
#define _JROOT_ERRSIZE 500
|
6
|
+
|
7
|
+
#define _JROOT_ROOT (void*)(1)
|
8
|
+
|
9
|
+
#define OUR_CONTEXT(js_context) \
|
10
|
+
({ \
|
11
|
+
JohnsonContext* _context; \
|
12
|
+
const VALUE _ruby_context = (VALUE)JS_GetContextPrivate(js_context); \
|
13
|
+
Data_Get_Struct(_ruby_context, JohnsonContext, _context); \
|
14
|
+
_context; \
|
15
|
+
})
|
16
|
+
|
17
|
+
#define OUR_RUNTIME(js_context) \
|
18
|
+
({ \
|
19
|
+
JohnsonRuntime* _johnson_runtime; \
|
20
|
+
JSRuntime * _js_runtime = JS_GetRuntime(js_context);\
|
21
|
+
const VALUE _ruby_runtime = (VALUE)JS_GetRuntimePrivate(_js_runtime); \
|
22
|
+
Data_Get_Struct(_ruby_runtime, JohnsonRuntime, _johnson_runtime); \
|
23
|
+
_johnson_runtime; \
|
24
|
+
})
|
25
|
+
|
26
|
+
|
27
|
+
#define _PREPARE_JROOTS(rb, context, cleancount) \
|
28
|
+
const bool _jroot_ruby = (rb); \
|
29
|
+
const int _jroot_cleans = (cleancount); \
|
30
|
+
void (*_jroot_cleanup[_jroot_cleans])(JSContext*, void*); \
|
31
|
+
void* _jroot_cleanup_data[_jroot_cleans]; \
|
32
|
+
JSContext* const _jroot_context = (context); \
|
33
|
+
int _jroot_cleanidx = 0;
|
34
|
+
|
35
|
+
#define PREPARE_JROOTS(context, cleancount) \
|
36
|
+
_PREPARE_JROOTS(false, context, cleancount)
|
37
|
+
|
38
|
+
#define PREPARE_RUBY_JROOTS(context, cleancount) \
|
39
|
+
_PREPARE_JROOTS(true, context, cleancount)
|
40
|
+
|
41
|
+
#define JCLEANUP(func, data) \
|
42
|
+
do \
|
43
|
+
{ \
|
44
|
+
assert(_jroot_cleanidx < _jroot_cleans); \
|
45
|
+
_jroot_cleanup[_jroot_cleanidx] = (func); \
|
46
|
+
_jroot_cleanup_data[_jroot_cleanidx] = (data); \
|
47
|
+
_jroot_cleanidx++; \
|
48
|
+
} while(0)
|
49
|
+
|
50
|
+
#define _JROOT(ptr, name) \
|
51
|
+
do \
|
52
|
+
{ \
|
53
|
+
static char _name[_JROOT_NAMESIZE] = ""; \
|
54
|
+
void* const _root = (ptr); \
|
55
|
+
if (*_name == '\0') \
|
56
|
+
snprintf(_name, _JROOT_NAMESIZE, "%s[%d]:%s: %s", __FILE__, __LINE__, __func__, (name)); \
|
57
|
+
JCHECK(JS_AddNamedRoot(_jroot_context, _root, _name)); \
|
58
|
+
JCLEANUP(_JROOT_ROOT, _root); \
|
59
|
+
} while(0)
|
60
|
+
|
61
|
+
#define JROOT(var) _JROOT(&(var), #var)
|
62
|
+
#define JROOT_PTR(ptr) _JROOT(ptr, #ptr)
|
63
|
+
|
64
|
+
#define JUNROOT(var) \
|
65
|
+
do \
|
66
|
+
{ \
|
67
|
+
void* const _jroot_match = &(var); \
|
68
|
+
int _jroot_i; \
|
69
|
+
for (_jroot_i = _jroot_cleanidx - 1; _jroot_i >= 0; _jroot_i--) \
|
70
|
+
if (_jroot_cleanup[_jroot_i] == _JROOT_ROOT && _jroot_cleanup_data[_jroot_i] == _jroot_match) \
|
71
|
+
{ \
|
72
|
+
JS_RemoveRoot(_jroot_context, _jroot_cleanup_data[_jroot_i]); \
|
73
|
+
if (_jroot_i == _jroot_cleanidx - 1) _jroot_cleanidx--; \
|
74
|
+
_jroot_cleanup[_jroot_i] = NULL; \
|
75
|
+
} \
|
76
|
+
} while (0)
|
77
|
+
|
78
|
+
#define REMOVE_JROOTS \
|
79
|
+
do \
|
80
|
+
{ \
|
81
|
+
int _jroot_i; \
|
82
|
+
for (_jroot_i = _jroot_cleanidx - 1; _jroot_i >= 0; _jroot_i--) \
|
83
|
+
{ \
|
84
|
+
if (_jroot_cleanup[_jroot_i] == _JROOT_ROOT) \
|
85
|
+
JS_RemoveRoot(_jroot_context, _jroot_cleanup_data[_jroot_i]); \
|
86
|
+
else if (_jroot_cleanup[_jroot_i]) \
|
87
|
+
(_jroot_cleanup[_jroot_i])(_jroot_context, _jroot_cleanup_data[_jroot_i]); \
|
88
|
+
} \
|
89
|
+
} while (0)
|
90
|
+
|
91
|
+
#define JCHECK_RUBY(cond) \
|
92
|
+
do \
|
93
|
+
{ \
|
94
|
+
assert(_jroot_ruby); \
|
95
|
+
if (!(cond)) \
|
96
|
+
{ \
|
97
|
+
REMOVE_JROOTS; \
|
98
|
+
raise_js_error_in_ruby(OUR_RUNTIME(_jroot_context)); \
|
99
|
+
} \
|
100
|
+
} while (0)
|
101
|
+
|
102
|
+
#define JCHECK(cond) \
|
103
|
+
do \
|
104
|
+
{ \
|
105
|
+
if (!(cond)) \
|
106
|
+
{ \
|
107
|
+
REMOVE_JROOTS; \
|
108
|
+
if (_jroot_ruby) \
|
109
|
+
raise_js_error_in_ruby(OUR_RUNTIME(_jroot_context)); \
|
110
|
+
else \
|
111
|
+
return JS_FALSE; \
|
112
|
+
} \
|
113
|
+
} while (0)
|
114
|
+
|
115
|
+
#define JPROTECT(func, data) \
|
116
|
+
({ \
|
117
|
+
int _state; \
|
118
|
+
const VALUE _old_errinfo = ruby_errinfo; \
|
119
|
+
const VALUE _result = rb_protect((func), (data), &_state); \
|
120
|
+
if (_state) \
|
121
|
+
{ \
|
122
|
+
REMOVE_JROOTS; \
|
123
|
+
if (_jroot_ruby) \
|
124
|
+
rb_jump_tag(_state); \
|
125
|
+
else \
|
126
|
+
return report_ruby_error_in_js(OUR_RUNTIME(_jroot_context), _state, _old_errinfo); \
|
127
|
+
} \
|
128
|
+
_result; \
|
129
|
+
})
|
130
|
+
|
131
|
+
#define JRETURN \
|
132
|
+
do \
|
133
|
+
{ \
|
134
|
+
assert(!_jroot_ruby); \
|
135
|
+
REMOVE_JROOTS; \
|
136
|
+
return JS_TRUE; \
|
137
|
+
} while(0)
|
138
|
+
|
139
|
+
#define JRETURN_RUBY(value) \
|
140
|
+
do \
|
141
|
+
{ \
|
142
|
+
assert(_jroot_ruby); \
|
143
|
+
typeof(value) _jroot_result = (value); \
|
144
|
+
REMOVE_JROOTS; \
|
145
|
+
return _jroot_result; \
|
146
|
+
} while(0)
|
147
|
+
|
148
|
+
#define JERROR(format, args...) \
|
149
|
+
do \
|
150
|
+
{ \
|
151
|
+
REMOVE_JROOTS; \
|
152
|
+
char _jroot_msg[_JROOT_ERRSIZE]; \
|
153
|
+
snprintf(_jroot_msg, _JROOT_ERRSIZE, (format) , ## args); \
|
154
|
+
if (_jroot_ruby) \
|
155
|
+
rb_raise(rb_eRuntimeError, _jroot_msg); \
|
156
|
+
else \
|
157
|
+
{ \
|
158
|
+
JSString* _jroot_err_str = JS_NewStringCopyZ(_jroot_context, _jroot_msg); \
|
159
|
+
if (_jroot_err_str) JS_SetPendingException(_jroot_context, STRING_TO_JSVAL(_jroot_err_str)); \
|
160
|
+
return JS_FALSE; \
|
161
|
+
} \
|
162
|
+
} while(0)
|
163
|
+
|
164
|
+
|
165
|
+
|
166
|
+
|
167
|
+
#define ARGLIST1(a) _data->a
|
168
|
+
#define ARGLIST2(a, b) _data->a, _data->b
|
169
|
+
#define ARGLIST3(a, b, c) _data->a, _data->b, _data->c
|
170
|
+
#define ARGLIST4(a, b, c, d) _data->a, _data->b, _data->c, _data->d
|
171
|
+
#define ARGLIST5(a, b, c, d, e) _data->a, _data->b, _data->c, _data->d, _data->e
|
172
|
+
|
173
|
+
#define DECLARE_RUBY_WRAPPER(name, args) \
|
174
|
+
typedef struct { args; } name ## _args; \
|
175
|
+
VALUE name ## _invoke(VALUE magic);
|
176
|
+
#define DEFINE_RUBY_WRAPPER(name, func, arglist) \
|
177
|
+
VALUE name ## _invoke(VALUE magic) \
|
178
|
+
{ \
|
179
|
+
name ## _args * _data = (name ## _args *)(FIX2INT(magic) << 2); \
|
180
|
+
return func(arglist); \
|
181
|
+
}
|
182
|
+
#define RUBY_WRAPPER_ARG(name, args...) ({ name ## _args _x = { args }; INT2FIX((int)(&_x) >> 2); })
|
183
|
+
#define RUBY_WRAPPER(name) name ## _invoke
|
184
|
+
#define CALL_RUBY_WRAPPER(name, args...) JPROTECT(RUBY_WRAPPER(name), RUBY_WRAPPER_ARG(name, args))
|
185
|
+
|
186
|
+
|
187
|
+
#endif
|
@@ -0,0 +1,609 @@
|
|
1
|
+
#include "js_land_proxy.h"
|
2
|
+
#include "conversions.h"
|
3
|
+
|
4
|
+
static JSBool get(JSContext* js_context, JSObject* obj, jsval id, jsval* retval);
|
5
|
+
static JSBool set(JSContext* context, JSObject* obj, jsval id, jsval* retval);
|
6
|
+
static JSBool construct(JSContext* js_context, JSObject* obj, uintN argc, jsval* argv, jsval* retval);
|
7
|
+
static JSBool resolve(JSContext *js_context, JSObject *obj, jsval id, uintN flags, JSObject **objp);
|
8
|
+
static JSBool call(JSContext* js_context, JSObject* obj, uintN argc, jsval* argv, jsval* retval);
|
9
|
+
static void finalize(JSContext* context, JSObject* obj);
|
10
|
+
|
11
|
+
static JSClass JSLandProxyClass = {
|
12
|
+
"JSLandProxy", JSCLASS_HAS_PRIVATE | JSCLASS_NEW_RESOLVE,
|
13
|
+
JS_PropertyStub,
|
14
|
+
JS_PropertyStub,
|
15
|
+
get,
|
16
|
+
set,
|
17
|
+
JS_EnumerateStub,
|
18
|
+
(JSResolveOp) resolve,
|
19
|
+
JS_ConvertStub,
|
20
|
+
finalize
|
21
|
+
};
|
22
|
+
|
23
|
+
static JSClass JSLandClassProxyClass = {
|
24
|
+
"JSLandClassProxy", JSCLASS_HAS_PRIVATE,
|
25
|
+
JS_PropertyStub,
|
26
|
+
JS_PropertyStub,
|
27
|
+
get,
|
28
|
+
set,
|
29
|
+
JS_EnumerateStub,
|
30
|
+
JS_ResolveStub,
|
31
|
+
JS_ConvertStub,
|
32
|
+
finalize,
|
33
|
+
NULL,
|
34
|
+
NULL,
|
35
|
+
NULL,
|
36
|
+
construct
|
37
|
+
};
|
38
|
+
|
39
|
+
static JSClass JSLandCallableProxyClass = {
|
40
|
+
"JSLandCallableProxy", JSCLASS_HAS_PRIVATE,
|
41
|
+
JS_PropertyStub,
|
42
|
+
JS_PropertyStub,
|
43
|
+
get,
|
44
|
+
set,
|
45
|
+
JS_EnumerateStub,
|
46
|
+
JS_ResolveStub,
|
47
|
+
JS_ConvertStub,
|
48
|
+
finalize,
|
49
|
+
NULL,
|
50
|
+
NULL,
|
51
|
+
call
|
52
|
+
};
|
53
|
+
|
54
|
+
static VALUE call_ruby_from_js_invoke(VALUE args)
|
55
|
+
{
|
56
|
+
VALUE self = rb_ary_pop(args);
|
57
|
+
VALUE id = rb_ary_pop(args);
|
58
|
+
return rb_apply(self, SYM2ID(id), args);
|
59
|
+
}
|
60
|
+
|
61
|
+
JSBool call_ruby_from_js_va(JohnsonRuntime* runtime, VALUE* result, VALUE self, ID id, int argc, va_list va)
|
62
|
+
{
|
63
|
+
VALUE old_errinfo = ruby_errinfo;
|
64
|
+
VALUE args = rb_ary_new2(argc + 2);
|
65
|
+
|
66
|
+
int i;
|
67
|
+
for(i = 0; i < argc; i++)
|
68
|
+
rb_ary_store(args, i, va_arg(va, VALUE));
|
69
|
+
|
70
|
+
rb_ary_store(args, argc, ID2SYM(id));
|
71
|
+
rb_ary_store(args, argc + 1, self);
|
72
|
+
|
73
|
+
int state;
|
74
|
+
*result = rb_protect(call_ruby_from_js_invoke, args, &state);
|
75
|
+
|
76
|
+
if (state)
|
77
|
+
return report_ruby_error_in_js(runtime, state, old_errinfo);
|
78
|
+
|
79
|
+
return JS_TRUE;
|
80
|
+
}
|
81
|
+
|
82
|
+
JSBool call_ruby_from_js(JohnsonRuntime* runtime, jsval* retval, VALUE self, ID id, int argc, ...)
|
83
|
+
{
|
84
|
+
VALUE result;
|
85
|
+
va_list va;
|
86
|
+
va_start(va, argc);
|
87
|
+
JSBool okay = call_ruby_from_js_va(runtime, &result, self, id, argc, va);
|
88
|
+
va_end(va);
|
89
|
+
if (!okay) return JS_FALSE;
|
90
|
+
return retval ? convert_to_js(runtime, result, retval) : JS_TRUE;
|
91
|
+
}
|
92
|
+
|
93
|
+
JSBool call_ruby_from_js2(JohnsonRuntime* runtime, VALUE* retval, VALUE self, ID id, int argc, ...)
|
94
|
+
{
|
95
|
+
va_list va;
|
96
|
+
va_start(va, argc);
|
97
|
+
JSBool okay = call_ruby_from_js_va(runtime, retval, self, id, argc, va);
|
98
|
+
va_end(va);
|
99
|
+
return okay;
|
100
|
+
}
|
101
|
+
|
102
|
+
static bool autovivified_p(VALUE UNUSED(ruby_context), VALUE self, char* name)
|
103
|
+
{
|
104
|
+
return RTEST(rb_funcall(Johnson_SpiderMonkey_JSLandProxy(), rb_intern("autovivified?"), 2,
|
105
|
+
self, rb_str_new2(name)));
|
106
|
+
}
|
107
|
+
|
108
|
+
static bool const_p(VALUE self, char* name)
|
109
|
+
{
|
110
|
+
return rb_obj_is_kind_of(self, rb_cModule)
|
111
|
+
&& rb_is_const_id(rb_intern(name))
|
112
|
+
&& RTEST( rb_funcall(self, rb_intern("const_defined?"), 1, ID2SYM(rb_intern(name))) );
|
113
|
+
}
|
114
|
+
|
115
|
+
static bool global_p(char* name)
|
116
|
+
{
|
117
|
+
return *name == '$' && rb_ary_includes(rb_f_global_variables(), rb_str_new2(name));
|
118
|
+
}
|
119
|
+
|
120
|
+
static bool method_p(VALUE self, char* name)
|
121
|
+
{
|
122
|
+
return RTEST( rb_funcall(self, rb_intern("respond_to?"), 1, ID2SYM(rb_intern(name))) );
|
123
|
+
}
|
124
|
+
|
125
|
+
static bool attribute_p(VALUE self, char* name)
|
126
|
+
{
|
127
|
+
if (!method_p(self, name))
|
128
|
+
return false;
|
129
|
+
|
130
|
+
VALUE rb_id = rb_intern(name);
|
131
|
+
VALUE rb_method = rb_funcall(self, rb_intern("method"), 1, ID2SYM(rb_id));
|
132
|
+
|
133
|
+
if (TYPE(rb_method) == T_DATA)
|
134
|
+
{
|
135
|
+
VALUE klass = CLASS_OF(rb_method);
|
136
|
+
if (klass == rb_cMethod)
|
137
|
+
{
|
138
|
+
METHOD* method;
|
139
|
+
Data_Get_Struct(rb_method, METHOD, method);
|
140
|
+
|
141
|
+
if (method && nd_type(method->body) == NODE_IVAR)
|
142
|
+
return true;
|
143
|
+
}
|
144
|
+
}
|
145
|
+
|
146
|
+
return RTEST(rb_funcall(Johnson_SpiderMonkey_JSLandProxy(),
|
147
|
+
rb_intern("js_property?"), 2, self, ID2SYM(rb_id)));
|
148
|
+
}
|
149
|
+
|
150
|
+
static bool indexable_p(VALUE self)
|
151
|
+
{
|
152
|
+
return RTEST(rb_funcall(self, rb_intern("respond_to?"), 1, ID2SYM(rb_intern("[]"))));
|
153
|
+
}
|
154
|
+
|
155
|
+
static bool has_key_p(VALUE self, char* name)
|
156
|
+
{
|
157
|
+
return RTEST(rb_funcall(self, rb_intern("respond_to?"), 1, ID2SYM(rb_intern("[]"))))
|
158
|
+
&& RTEST(rb_funcall(self, rb_intern("respond_to?"), 1, ID2SYM(rb_intern("key?"))))
|
159
|
+
&& RTEST(rb_funcall(self, rb_intern("key?"), 1, rb_str_new2(name)));
|
160
|
+
}
|
161
|
+
|
162
|
+
static bool respond_to_p(JSContext* js_context, JSObject* obj, char* name)
|
163
|
+
{
|
164
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
165
|
+
|
166
|
+
JohnsonContext* context;
|
167
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
168
|
+
|
169
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(
|
170
|
+
context->js, obj, JS_GET_CLASS(context->js, obj), NULL);
|
171
|
+
|
172
|
+
if (!self) return false;
|
173
|
+
|
174
|
+
return autovivified_p(ruby_context, self, name)
|
175
|
+
|| const_p(self, name)
|
176
|
+
|| global_p(name)
|
177
|
+
|| attribute_p(self, name)
|
178
|
+
|| method_p(self, name)
|
179
|
+
|| has_key_p(self, name);
|
180
|
+
}
|
181
|
+
|
182
|
+
static jsval evaluate_js_property_expression(JohnsonRuntime * runtime, const char * property, jsval* retval) {
|
183
|
+
JSContext * context = johnson_get_current_context(runtime);
|
184
|
+
return JS_EvaluateScript(context, runtime->global,
|
185
|
+
property, strlen(property), "johnson:evaluate_js_property_expression", 1,
|
186
|
+
retval);
|
187
|
+
}
|
188
|
+
|
189
|
+
static JSBool get(JSContext* js_context, JSObject* obj, jsval id, jsval* retval)
|
190
|
+
{
|
191
|
+
// pull out our Ruby context, which is embedded in js_context
|
192
|
+
|
193
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
194
|
+
|
195
|
+
// get our struct, which is embedded in ruby_context
|
196
|
+
|
197
|
+
JohnsonContext* context;
|
198
|
+
JohnsonRuntime* runtime;
|
199
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
200
|
+
|
201
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
202
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
203
|
+
|
204
|
+
PREPARE_JROOTS(js_context, 1);
|
205
|
+
JROOT(id);
|
206
|
+
|
207
|
+
// get the Ruby object that backs this proxy
|
208
|
+
|
209
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, obj, JS_GET_CLASS(context->js, obj), NULL);
|
210
|
+
|
211
|
+
// Short-circuit for numeric indexes
|
212
|
+
|
213
|
+
if (JSVAL_IS_INT(id))
|
214
|
+
{
|
215
|
+
if (indexable_p(self)) {
|
216
|
+
VALUE idx = INT2FIX(JSVAL_TO_INT(id));
|
217
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, rb_intern("[]"), 1, idx));
|
218
|
+
}
|
219
|
+
|
220
|
+
JRETURN;
|
221
|
+
}
|
222
|
+
|
223
|
+
char* name = JS_GetStringBytes(JSVAL_TO_STRING(id));
|
224
|
+
VALUE ruby_id = rb_intern(name);
|
225
|
+
|
226
|
+
// FIXME: we should probably just JS_DefineProperty this, and it shouldn't be enumerable
|
227
|
+
|
228
|
+
if (!strcasecmp("__iterator__", name)) {
|
229
|
+
JCHECK(evaluate_js_property_expression(runtime, "Johnson.Generator.create", retval));
|
230
|
+
}
|
231
|
+
|
232
|
+
// if the Ruby object has a dynamic js property with a key
|
233
|
+
// matching the property we're looking for, pull the value out of
|
234
|
+
// that map.
|
235
|
+
|
236
|
+
else if (autovivified_p(ruby_context, self, name))
|
237
|
+
{
|
238
|
+
JCHECK(call_ruby_from_js(runtime, retval, Johnson_SpiderMonkey_JSLandProxy(),
|
239
|
+
rb_intern("autovivified"), 2, self, rb_str_new2(name)));
|
240
|
+
}
|
241
|
+
|
242
|
+
// if the Ruby object is a Module or Class and has a matching
|
243
|
+
// const defined, return the converted result of const_get
|
244
|
+
|
245
|
+
else if (const_p(self, name))
|
246
|
+
{
|
247
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, rb_intern("const_get"),
|
248
|
+
1, ID2SYM(ruby_id)));
|
249
|
+
}
|
250
|
+
|
251
|
+
// otherwise, if it's a global, return the global
|
252
|
+
else if (global_p(name))
|
253
|
+
{
|
254
|
+
JCHECK(convert_to_js(runtime, rb_gv_get(name), retval));
|
255
|
+
}
|
256
|
+
|
257
|
+
// otherwise, if the Ruby object has a an attribute method matching
|
258
|
+
// the property we're trying to get, call it and return the converted result
|
259
|
+
|
260
|
+
else if (attribute_p(self, name))
|
261
|
+
{
|
262
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, ruby_id, 0));
|
263
|
+
}
|
264
|
+
|
265
|
+
// otherwise, if the Ruby object quacks sorta like a hash (it responds to
|
266
|
+
// "[]" and "key?"), index it by key and return the converted result
|
267
|
+
|
268
|
+
else if (has_key_p(self, name))
|
269
|
+
{
|
270
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, rb_intern("[]"), 1, rb_str_new2(name)));
|
271
|
+
}
|
272
|
+
|
273
|
+
// otherwise, it's a method being accessed as a property, which means
|
274
|
+
// we need to return a lambda
|
275
|
+
|
276
|
+
// FIXME: this should really wrap the Method for 'name' in a JS class
|
277
|
+
// rather than generating a wrapper Proc
|
278
|
+
|
279
|
+
else if (method_p(self, name))
|
280
|
+
{
|
281
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, rb_intern("method"), 1, rb_str_new2(name)));
|
282
|
+
}
|
283
|
+
|
284
|
+
// else it's undefined (JS_VOID) by default
|
285
|
+
JRETURN;
|
286
|
+
}
|
287
|
+
|
288
|
+
// called for lazily resolved properties, which should go away
|
289
|
+
static JSBool get_and_destroy_resolved_property(
|
290
|
+
JSContext* js_context, JSObject* obj, jsval id, jsval* retval)
|
291
|
+
{
|
292
|
+
PREPARE_JROOTS(js_context, 1);
|
293
|
+
JROOT(id);
|
294
|
+
char* name = JS_GetStringBytes(JSVAL_TO_STRING(id));
|
295
|
+
JCHECK(JS_DeleteProperty(js_context, obj, name));
|
296
|
+
JCHECK(get(js_context, obj, id, retval));
|
297
|
+
JRETURN;
|
298
|
+
}
|
299
|
+
|
300
|
+
static JSBool set(JSContext* js_context, JSObject* obj, jsval id, jsval* value)
|
301
|
+
{
|
302
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
303
|
+
|
304
|
+
JohnsonContext* context;
|
305
|
+
JohnsonRuntime* runtime;
|
306
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
307
|
+
|
308
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
309
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
310
|
+
|
311
|
+
PREPARE_JROOTS(js_context, 2);
|
312
|
+
JROOT(id);
|
313
|
+
JROOT_PTR(value);
|
314
|
+
|
315
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, obj, JS_GET_CLASS(context->js, obj), NULL);
|
316
|
+
|
317
|
+
// Short-circuit for numeric indexes
|
318
|
+
|
319
|
+
if (JSVAL_IS_INT(id))
|
320
|
+
{
|
321
|
+
if (indexable_p(self))
|
322
|
+
{
|
323
|
+
VALUE idx = INT2FIX(JSVAL_TO_INT(id));
|
324
|
+
VALUE val = CONVERT_TO_RUBY(runtime, *value);
|
325
|
+
|
326
|
+
JCHECK(call_ruby_from_js(runtime, NULL, self, rb_intern("[]="), 2, idx, val));
|
327
|
+
}
|
328
|
+
|
329
|
+
JRETURN;
|
330
|
+
}
|
331
|
+
|
332
|
+
VALUE ruby_key = CONVERT_TO_RUBY(runtime, id);
|
333
|
+
VALUE ruby_value = CONVERT_TO_RUBY(runtime, *value);
|
334
|
+
|
335
|
+
VALUE setter = rb_str_append(rb_str_new3(ruby_key), rb_str_new2("="));
|
336
|
+
VALUE setter_id = rb_intern(StringValueCStr(setter));
|
337
|
+
|
338
|
+
VALUE settable_p, indexable_p;
|
339
|
+
JCHECK(call_ruby_from_js2(runtime, &settable_p, self, rb_intern("respond_to?"), 1, ID2SYM(setter_id)));
|
340
|
+
JCHECK(call_ruby_from_js2(runtime, &indexable_p, self, rb_intern("respond_to?"), 1, ID2SYM(rb_intern("[]="))));
|
341
|
+
|
342
|
+
if (settable_p)
|
343
|
+
{
|
344
|
+
VALUE method, arity;
|
345
|
+
JCHECK(call_ruby_from_js2(runtime, &method, self, rb_intern("method"), 1, ID2SYM(setter_id)));
|
346
|
+
JCHECK(call_ruby_from_js2(runtime, &arity, method, rb_intern("arity"), 0));
|
347
|
+
|
348
|
+
// if the Ruby object has a 1-arity method named "property=",
|
349
|
+
// call it with the converted value
|
350
|
+
|
351
|
+
if (NUM2INT(arity) == 1)
|
352
|
+
JCHECK(call_ruby_from_js(runtime, NULL, self, setter_id, 1, ruby_value));
|
353
|
+
}
|
354
|
+
else if(indexable_p)
|
355
|
+
{
|
356
|
+
// otherwise, if the Ruby object quacks sorta like a hash for assignment
|
357
|
+
// (it responds to "[]="), assign it by key
|
358
|
+
|
359
|
+
JCHECK(call_ruby_from_js(runtime, NULL, self, rb_intern("[]="), 2, ruby_key, ruby_value));
|
360
|
+
}
|
361
|
+
else
|
362
|
+
{
|
363
|
+
JCHECK(call_ruby_from_js(runtime, NULL, Johnson_SpiderMonkey_JSLandProxy(), rb_intern("autovivify"),
|
364
|
+
3, self, ruby_key, ruby_value));
|
365
|
+
}
|
366
|
+
|
367
|
+
JRETURN;
|
368
|
+
}
|
369
|
+
|
370
|
+
static JSBool construct(JSContext* js_context, JSObject* UNUSED(obj), uintN argc, jsval* argv, jsval* retval)
|
371
|
+
{
|
372
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
373
|
+
|
374
|
+
JohnsonContext* context;
|
375
|
+
JohnsonRuntime* runtime;
|
376
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
377
|
+
|
378
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
379
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
380
|
+
|
381
|
+
PREPARE_JROOTS(js_context, 0);
|
382
|
+
|
383
|
+
VALUE klass = CONVERT_TO_RUBY(runtime, JS_ARGV_CALLEE(argv));
|
384
|
+
VALUE args = rb_ary_new();
|
385
|
+
|
386
|
+
uintN i;
|
387
|
+
for (i = 0; i < argc; ++i)
|
388
|
+
rb_ary_push(args, CONVERT_TO_RUBY(runtime, argv[i]));
|
389
|
+
|
390
|
+
JCHECK(call_ruby_from_js(runtime, retval, Johnson_SpiderMonkey_JSLandProxy(),
|
391
|
+
rb_intern("send_with_possible_block"), 3, klass, ID2SYM(rb_intern("new")), args));
|
392
|
+
JRETURN;
|
393
|
+
}
|
394
|
+
|
395
|
+
static JSBool resolve(JSContext *js_context, JSObject *obj, jsval id, uintN UNUSED(flags), JSObject **objp)
|
396
|
+
{
|
397
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
398
|
+
|
399
|
+
JohnsonContext* context;
|
400
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
401
|
+
|
402
|
+
PREPARE_JROOTS(js_context, 1);
|
403
|
+
JROOT(id);
|
404
|
+
|
405
|
+
char* name = JS_GetStringBytes(JS_ValueToString(js_context, id));
|
406
|
+
|
407
|
+
if (respond_to_p(js_context, obj, name))
|
408
|
+
{
|
409
|
+
JCHECK(JS_DefineProperty(js_context, obj, name, JSVAL_VOID,
|
410
|
+
get_and_destroy_resolved_property, set, JSPROP_ENUMERATE));
|
411
|
+
|
412
|
+
*objp = obj;
|
413
|
+
}
|
414
|
+
|
415
|
+
JRETURN;
|
416
|
+
}
|
417
|
+
|
418
|
+
static JSBool to_string(JSContext* js_context, JSObject* obj, uintN UNUSED(argc), jsval* UNUSED(argv), jsval* retval)
|
419
|
+
{
|
420
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
421
|
+
|
422
|
+
JohnsonContext* context;
|
423
|
+
JohnsonRuntime* runtime;
|
424
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
425
|
+
|
426
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
427
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
428
|
+
|
429
|
+
PREPARE_JROOTS(js_context, 0);
|
430
|
+
|
431
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, obj, JS_GET_CLASS(context->js, obj), NULL);
|
432
|
+
|
433
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, rb_intern("to_s"), 0));
|
434
|
+
JRETURN;
|
435
|
+
}
|
436
|
+
|
437
|
+
static JSBool to_array(JSContext* js_context, JSObject* obj, uintN UNUSED(argc), jsval* UNUSED(argv), jsval* retval)
|
438
|
+
{
|
439
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
440
|
+
|
441
|
+
JohnsonContext* context;
|
442
|
+
JohnsonRuntime* runtime;
|
443
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
444
|
+
|
445
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
446
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
447
|
+
|
448
|
+
PREPARE_JROOTS(js_context, 0);
|
449
|
+
|
450
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, obj, JS_GET_CLASS(context->js, obj), NULL);
|
451
|
+
|
452
|
+
JCHECK(call_ruby_from_js(runtime, retval, self, rb_intern("to_a"), 0));
|
453
|
+
JRETURN;
|
454
|
+
}
|
455
|
+
|
456
|
+
static JSBool method_missing(JSContext* js_context, JSObject* obj, uintN argc, jsval* argv, jsval* retval)
|
457
|
+
{
|
458
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
459
|
+
|
460
|
+
JohnsonContext* context;
|
461
|
+
JohnsonRuntime* runtime;
|
462
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
463
|
+
|
464
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
465
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
466
|
+
|
467
|
+
PREPARE_JROOTS(js_context, 0);
|
468
|
+
|
469
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, obj, JS_GET_CLASS(context->js, obj), NULL);
|
470
|
+
|
471
|
+
assert(argc >= 2);
|
472
|
+
|
473
|
+
char* key = JS_GetStringBytes(JSVAL_TO_STRING(argv[0]));
|
474
|
+
VALUE ruby_id = rb_intern(key);
|
475
|
+
|
476
|
+
// FIXME: this is horrible and lazy, to_a comes from enumerable on proxy (argv[1] is a JSArray)
|
477
|
+
VALUE args;
|
478
|
+
JCHECK(call_ruby_from_js2(runtime, &args, CONVERT_TO_RUBY(runtime, argv[1]), rb_intern("to_a"), 0));
|
479
|
+
|
480
|
+
JCHECK(call_ruby_from_js(runtime, retval, Johnson_SpiderMonkey_JSLandProxy(),
|
481
|
+
rb_intern("send_with_possible_block"), 3, self, ID2SYM(ruby_id), args));
|
482
|
+
|
483
|
+
JRETURN;
|
484
|
+
}
|
485
|
+
|
486
|
+
static JSBool call(JSContext* js_context, JSObject* UNUSED(obj), uintN argc, jsval* argv, jsval* retval)
|
487
|
+
{
|
488
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
489
|
+
|
490
|
+
JohnsonContext* context;
|
491
|
+
JohnsonRuntime* runtime;
|
492
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
493
|
+
|
494
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
495
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
496
|
+
|
497
|
+
PREPARE_JROOTS(js_context, 0);
|
498
|
+
|
499
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, JSVAL_TO_OBJECT(JS_ARGV_CALLEE(argv)), &JSLandCallableProxyClass, NULL);
|
500
|
+
|
501
|
+
VALUE args = rb_ary_new();
|
502
|
+
|
503
|
+
uintN i;
|
504
|
+
for (i = 0; i < argc; ++i)
|
505
|
+
rb_ary_push(args, CONVERT_TO_RUBY(runtime, argv[i]));
|
506
|
+
|
507
|
+
JCHECK(call_ruby_from_js(runtime, retval, Johnson_SpiderMonkey_JSLandProxy(),
|
508
|
+
rb_intern("send_with_possible_block"), 3, self, ID2SYM(rb_intern("call")), args));
|
509
|
+
JRETURN;
|
510
|
+
}
|
511
|
+
|
512
|
+
bool js_value_is_proxy(JohnsonRuntime* MAYBE_UNUSED(runtime), jsval maybe_proxy)
|
513
|
+
{
|
514
|
+
JSClass* klass = JS_GET_CLASS(
|
515
|
+
johnson_get_current_context(runtime),
|
516
|
+
JSVAL_TO_OBJECT(maybe_proxy));
|
517
|
+
|
518
|
+
return &JSLandProxyClass == klass
|
519
|
+
|| &JSLandClassProxyClass == klass
|
520
|
+
|| &JSLandCallableProxyClass == klass;
|
521
|
+
}
|
522
|
+
|
523
|
+
VALUE unwrap_js_land_proxy(JohnsonRuntime* runtime, jsval proxy)
|
524
|
+
{
|
525
|
+
VALUE value;
|
526
|
+
JSObject *proxy_object = JSVAL_TO_OBJECT(proxy);
|
527
|
+
JSContext * context = johnson_get_current_context(runtime);
|
528
|
+
|
529
|
+
value = (VALUE)JS_GetInstancePrivate(context, proxy_object,
|
530
|
+
JS_GET_CLASS(context, proxy_object), NULL);
|
531
|
+
|
532
|
+
return value;
|
533
|
+
}
|
534
|
+
|
535
|
+
static void finalize(JSContext* js_context, JSObject* obj)
|
536
|
+
{
|
537
|
+
VALUE ruby_context = (VALUE)JS_GetContextPrivate(js_context);
|
538
|
+
|
539
|
+
if (ruby_context)
|
540
|
+
{
|
541
|
+
JohnsonContext* context;
|
542
|
+
JohnsonRuntime* runtime;
|
543
|
+
Data_Get_Struct(ruby_context, JohnsonContext, context);
|
544
|
+
|
545
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(JS_GetRuntime(js_context));
|
546
|
+
Data_Get_Struct(ruby_runtime, JohnsonRuntime, runtime);
|
547
|
+
|
548
|
+
VALUE self = (VALUE)JS_GetInstancePrivate(context->js, obj,
|
549
|
+
JS_GET_CLASS(context->js, obj), NULL);
|
550
|
+
|
551
|
+
// remove the proxy OID from the id map
|
552
|
+
JS_HashTableRemove(runtime->rbids, (void *)self);
|
553
|
+
|
554
|
+
// free up the ruby value for GC
|
555
|
+
call_ruby_from_js(runtime, NULL, ruby_runtime, rb_intern("remove_gcthing"), 1, self);
|
556
|
+
}
|
557
|
+
}
|
558
|
+
|
559
|
+
JSBool make_js_land_proxy(JohnsonRuntime* runtime, VALUE value, jsval* retval)
|
560
|
+
{
|
561
|
+
*retval = (jsval)JS_HashTableLookup(runtime->rbids, (void *)value);
|
562
|
+
|
563
|
+
if (*retval)
|
564
|
+
{
|
565
|
+
return JS_TRUE;
|
566
|
+
}
|
567
|
+
else
|
568
|
+
{
|
569
|
+
JSContext * context = johnson_get_current_context(runtime);
|
570
|
+
PREPARE_JROOTS(context, 1);
|
571
|
+
|
572
|
+
JSObject *jsobj;
|
573
|
+
|
574
|
+
JSClass *klass = &JSLandProxyClass;
|
575
|
+
if (T_CLASS == TYPE(value)) klass = &JSLandClassProxyClass;
|
576
|
+
|
577
|
+
// FIXME: hack; should happen in Rubyland
|
578
|
+
if (T_STRUCT == TYPE(value))
|
579
|
+
rb_funcall(Johnson_SpiderMonkey_JSLandProxy(),
|
580
|
+
rb_intern("treat_all_properties_as_methods"), 1, value);
|
581
|
+
|
582
|
+
bool callable_p = Qtrue == rb_funcall(value,
|
583
|
+
rb_intern("respond_to?"), 1, rb_str_new2("call"));
|
584
|
+
|
585
|
+
if (callable_p)
|
586
|
+
klass = &JSLandCallableProxyClass;
|
587
|
+
|
588
|
+
JCHECK((jsobj = JS_NewObject(context, klass, NULL, NULL)));
|
589
|
+
JROOT(jsobj);
|
590
|
+
|
591
|
+
JCHECK(JS_SetPrivate(context, jsobj, (void*)value));
|
592
|
+
|
593
|
+
JCHECK(JS_DefineFunction(context, jsobj, "__noSuchMethod__", method_missing, 2, 0));
|
594
|
+
|
595
|
+
JCHECK(JS_DefineFunction(context, jsobj, "toArray", to_array, 0, 0));
|
596
|
+
JCHECK(JS_DefineFunction(context, jsobj, "toString", to_string, 0, 0));
|
597
|
+
|
598
|
+
*retval = OBJECT_TO_JSVAL(jsobj);
|
599
|
+
|
600
|
+
// put the proxy OID in the id map
|
601
|
+
JCHECK(JS_HashTableAdd(runtime->rbids, (void *)value, (void *)(*retval)));
|
602
|
+
|
603
|
+
// root the ruby value for GC
|
604
|
+
VALUE ruby_runtime = (VALUE)JS_GetRuntimePrivate(runtime->js);
|
605
|
+
rb_funcall(ruby_runtime, rb_intern("add_gcthing"), 1, value);
|
606
|
+
|
607
|
+
JRETURN;
|
608
|
+
}
|
609
|
+
}
|