/* Copyright JS Foundation and other contributors, http://js.foundation * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #include "jerryscript.h" #include "config.h" #include "test-common.h" /** Test in Proxy on C side. Equivalent test code in JS: var demo = 0.0; var target = {}; var handler = { get: function (target, name, recv) { assert (typeof (target) === 'object'); assert (name === 'value'); assert (typeof (recv) === 'object'); return demo++; } set: function (target, name, value, recv) { assert (typeof (target) === 'object'); assert (name === 'value'); assert (typeof (recv) === 'object'); demo = 55; return demo; } }; var pdemo = new Proxy(target, handler); assert (pdemo.value === 1.0); assert (pdemo.value === 1.0); assert (pdemo.value === 2.0); pdemo.value = 55; assert (pdemo.value === 56); pdemo.value = 12; assert (pdemo.value === 13); */ static int demo_value = 0; static jerry_value_t handler_get (const jerry_call_info_t *call_info_p, /**< call information */ const jerry_value_t args_p[], /**< function arguments */ const jerry_length_t args_count) /**< number of function arguments */ { JERRY_UNUSED (call_info_p); TEST_ASSERT (args_count == 3); TEST_ASSERT (jerry_value_is_object (args_p[0])); /* target */ TEST_ASSERT (jerry_value_is_string (args_p[1])); /* P */ TEST_ASSERT (jerry_value_is_object (args_p[2])); /* receiver */ const char expected[] = "value"; char buffer[10]; jerry_size_t copied = jerry_string_to_buffer (args_p[1], JERRY_ENCODING_CESU8, (jerry_char_t *) buffer, 10); TEST_ASSERT (copied == 5); TEST_ASSERT (strncmp (expected, buffer, 5) == 0); demo_value++; return jerry_number (demo_value); } /* handler_get */ static jerry_value_t handler_set (const jerry_call_info_t *call_info_p, /**< call information */ const jerry_value_t args_p[], /**< function arguments */ const jerry_length_t args_count) /**< number of function arguments */ { JERRY_UNUSED (call_info_p); JERRY_UNUSED (args_p); JERRY_UNUSED (args_count); TEST_ASSERT (args_count == 4); TEST_ASSERT (jerry_value_is_object (args_p[0])); /* target */ TEST_ASSERT (jerry_value_is_string (args_p[1])); /* P */ TEST_ASSERT (jerry_value_is_number (args_p[2])); /* V */ TEST_ASSERT (jerry_value_is_object (args_p[3])); /* receiver */ const char expected[] = "value"; char buffer[10]; jerry_size_t copied = jerry_string_to_buffer (args_p[1], JERRY_ENCODING_CESU8, (jerry_char_t *) buffer, 10); TEST_ASSERT (copied == 5); TEST_ASSERT (strncmp (expected, buffer, 5) == 0); TEST_ASSERT (jerry_value_is_number (args_p[2])); demo_value = (int) jerry_value_as_number (args_p[2]); return jerry_number (demo_value); } /* handler_set */ static void set_property (jerry_value_t target, /**< target object */ const char *name_p, /**< name of the property */ jerry_value_t value) /**< value of the property */ { jerry_value_t name_val = jerry_string_sz (name_p); jerry_value_t result_val = jerry_object_set (target, name_val, value); TEST_ASSERT (jerry_value_is_boolean (result_val)); TEST_ASSERT (jerry_value_is_true (result_val)); jerry_value_free (name_val); } /* set_property */ static jerry_value_t get_property (jerry_value_t target, /**< target object */ const char *name_p) /**< name of the property */ { jerry_value_t name_val = jerry_string_sz (name_p); jerry_value_t result_val = jerry_object_get (target, name_val); TEST_ASSERT (!jerry_value_is_exception (result_val)); jerry_value_free (name_val); return result_val; } /* get_property */ static void set_function (jerry_value_t target, /**< target object */ const char *name_p, /**< name of the function */ jerry_external_handler_t handler_p) /**< function callback */ { jerry_value_t function_val = jerry_function_external (handler_p); set_property (target, name_p, function_val); jerry_value_free (function_val); } /* set_function */ struct test_data { int value; }; static void proxy_native_freecb (void *native_p, /**< native pointer */ jerry_object_native_info_t *info_p) /**< native info */ { TEST_ASSERT (native_p != NULL); TEST_ASSERT (info_p->free_cb == proxy_native_freecb); struct test_data *data_p = (struct test_data *) native_p; data_p->value = -1; } /* proxy_native_freecb */ static const jerry_object_native_info_t proxy_native_info = { .free_cb = proxy_native_freecb, .number_of_references = 0, .offset_of_references = 0, }; static jerry_value_t proxy_native_handler_get (const jerry_call_info_t *call_info_p, /**< call information */ const jerry_value_t args_p[], /**< function arguments */ const jerry_length_t args_count) /**< number of function arguments */ { JERRY_UNUSED (call_info_p); TEST_ASSERT (args_count == 3); /* 3rd argument (Receiver) should be the Proxy here. */ jerry_value_t receiver = args_p[2]; TEST_ASSERT (jerry_value_is_proxy (receiver)); /* Check if proxy has the native ptr. */ TEST_ASSERT (jerry_object_has_native_ptr (receiver, &proxy_native_info)); struct test_data *native_p = jerry_object_get_native_ptr (receiver, &proxy_native_info); TEST_ASSERT (native_p != NULL); native_p->value <<= 1; return jerry_number (native_p->value); } /* proxy_native_handler_get */ /** * Test Proxy with added native object. */ static void test_proxy_native (void) { jerry_value_t handler = jerry_object (); set_function (handler, "get", proxy_native_handler_get); jerry_value_t target = jerry_object (); jerry_value_t proxy = jerry_proxy (target, handler); struct test_data *data = (struct test_data *) malloc (sizeof (struct test_data)); data->value = 2; jerry_object_set_native_ptr (proxy, &proxy_native_info, data); /* Call: proxy[10] */ jerry_value_t result_for_10 = jerry_object_get_index (proxy, 10); TEST_ASSERT (jerry_value_is_number (result_for_10)); TEST_ASSERT (jerry_value_as_number (result_for_10) == 4.0); /* Call: proxy[5] */ data->value = 8; jerry_value_t result_for_5 = jerry_object_get_index (proxy, 5); TEST_ASSERT (jerry_value_is_number (result_for_5)); TEST_ASSERT (jerry_value_as_number (result_for_5) == 16.0); jerry_value_free (handler); jerry_value_free (target); jerry_value_free (proxy); } /* test_proxy_native */ int main (void) { TEST_INIT (); if (!jerry_feature_enabled (JERRY_FEATURE_PROXY)) { printf ("Skipping test, Proxy not enabled\n"); return 0; } jerry_init (JERRY_INIT_EMPTY); jerry_value_t handler = jerry_object (); { set_function (handler, "get", handler_get); set_function (handler, "set", handler_set); } jerry_value_t target = jerry_object (); jerry_value_t proxy = jerry_proxy (target, handler); { jerry_value_t global = jerry_current_realm (); set_property (global, "pdemo", proxy); jerry_value_free (global); } const jerry_char_t get_value_src[] = TEST_STRING_LITERAL ("pdemo.value"); jerry_value_t parsed_get_code_val = jerry_parse (get_value_src, sizeof (get_value_src) - 1, NULL); TEST_ASSERT (!jerry_value_is_exception (parsed_get_code_val)); { jerry_value_t res = jerry_run (parsed_get_code_val); TEST_ASSERT (jerry_value_is_number (res)); TEST_ASSERT (jerry_value_as_number (res) == 1.0); jerry_value_free (res); } { jerry_value_t res = get_property (proxy, "value"); TEST_ASSERT (jerry_value_is_number (res)); TEST_ASSERT (jerry_value_as_number (res) == 2.0); jerry_value_free (res); } { jerry_value_t res = jerry_run (parsed_get_code_val); TEST_ASSERT (jerry_value_is_number (res)); TEST_ASSERT (jerry_value_as_number (res) == 3.0); jerry_value_free (res); } const jerry_char_t set_value_src[] = TEST_STRING_LITERAL ("pdemo.value = 55"); jerry_value_t parsed_set_code_val = jerry_parse (set_value_src, sizeof (set_value_src) - 1, NULL); TEST_ASSERT (!jerry_value_is_exception (parsed_set_code_val)); { jerry_value_t res = jerry_run (parsed_set_code_val); TEST_ASSERT (jerry_value_is_number (res)); TEST_ASSERT (jerry_value_as_number (res) == 55); jerry_value_free (res); } { jerry_value_t res = jerry_run (parsed_get_code_val); TEST_ASSERT (jerry_value_is_number (res)); TEST_ASSERT (jerry_value_as_number (res) == 56); jerry_value_free (res); } { jerry_value_t new_value = jerry_number (12); set_property (proxy, "value", new_value); jerry_value_free (new_value); } { jerry_value_t res = get_property (proxy, "value"); TEST_ASSERT (jerry_value_is_number (res)); TEST_ASSERT (jerry_value_as_number (res) == 13.0); jerry_value_free (res); } jerry_value_free (parsed_set_code_val); jerry_value_free (parsed_get_code_val); jerry_value_free (proxy); jerry_value_free (target); jerry_value_free (handler); { const jerry_char_t has_value_src[] = TEST_STRING_LITERAL ("new Proxy({}, {\n" " has: function(target, key) { throw 33 }\n" "})"); jerry_value_t parsed_has_code_val = jerry_parse (has_value_src, sizeof (has_value_src) - 1, NULL); TEST_ASSERT (!jerry_value_is_exception (parsed_has_code_val)); jerry_value_t res = jerry_run (parsed_has_code_val); jerry_value_free (parsed_has_code_val); TEST_ASSERT (jerry_value_is_proxy (res)); jerry_value_t name = jerry_string_sz ("key"); TEST_ASSERT (jerry_value_is_string (name)); jerry_value_t property = jerry_object_has (res, name); jerry_value_free (name); jerry_value_free (res); TEST_ASSERT (jerry_value_is_exception (property)); property = jerry_exception_value (property, true); TEST_ASSERT (jerry_value_as_number (property) == 33); jerry_value_free (property); } target = jerry_object (); handler = jerry_object (); proxy = jerry_proxy (target, handler); { jerry_value_t res = jerry_proxy_target (proxy); TEST_ASSERT (res == target); jerry_value_free (res); res = jerry_proxy_handler (proxy); TEST_ASSERT (res == handler); jerry_value_free (res); res = jerry_proxy_target (target); TEST_ASSERT (jerry_value_is_exception (res)); res = jerry_exception_value (res, true); TEST_ASSERT (jerry_error_type (res) == JERRY_ERROR_TYPE); jerry_value_free (res); res = jerry_proxy_handler (handler); TEST_ASSERT (jerry_value_is_exception (res)); res = jerry_exception_value (res, true); TEST_ASSERT (jerry_error_type (res) == JERRY_ERROR_TYPE); jerry_value_free (res); } jerry_value_free (proxy); jerry_value_free (handler); jerry_value_free (target); test_proxy_native (); jerry_cleanup (); return 0; } /* main */