Skip to content

Commit 100f895

Browse files
committed
Introduce new Promise API methods
The new API methods make it possible to get a Promise object's result and it's state. JerryScript-DCO-1.0-Signed-off-by: Peter Gal pgal.u-szeged@partner.samsung.com
1 parent a8fb4ee commit 100f895

File tree

5 files changed

+253
-3
lines changed

5 files changed

+253
-3
lines changed

jerry-core/api/jerry.c

Lines changed: 59 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,14 @@ JERRY_STATIC_ASSERT ((int) RE_FLAG_GLOBAL == (int) JERRY_REGEXP_FLAG_GLOBAL
7171
re_flags_t_must_be_equal_to_jerry_regexp_flags_t);
7272
#endif /* ENABLED (JERRY_BUILTIN_REGEXP) */
7373

74+
#if ENABLED (JERRY_ES2015_BUILTIN_PROMISE)
75+
/* The internal ECMA_PROMISE_STATE_* values are "one byte away" from the API values */
76+
JERRY_STATIC_ASSERT (((ECMA_PROMISE_STATE_PENDING + 1) == JERRY_PROMISE_STATE_PENDING)
77+
&& ((ECMA_PROMISE_STATE_FULFILLED + 1) == JERRY_PROMISE_STATE_FULFILLED)
78+
&& ((ECMA_PROMISE_STATE_REJECTED + 1) == JERRY_PROMISE_STATE_REJECTED),
79+
promise_internal_state_matches_external);
80+
#endif /* ENABLED (JERRY_ES2015_BUILTIN_PROMISE) */
81+
7482
#if !ENABLED (JERRY_PARSER) && !ENABLED (JERRY_SNAPSHOT_EXEC)
7583
#error "JERRY_SNAPSHOT_EXEC must be enabled if JERRY_PARSER is disabled!"
7684
#endif /* !ENABLED (JERRY_PARSER) && !ENABLED (JERRY_SNAPSHOT_EXEC) */
@@ -2845,6 +2853,57 @@ jerry_resolve_or_reject_promise (jerry_value_t promise, /**< the promise value *
28452853
#endif /* ENABLED (JERRY_ES2015_BUILTIN_PROMISE) */
28462854
} /* jerry_resolve_or_reject_promise */
28472855

2856+
/**
2857+
* Get the result of a promise.
2858+
*
2859+
* @return - Promise result
2860+
* - Type error if the promise support was not enabled or the input was not a promise object
2861+
*/
2862+
jerry_value_t
2863+
jerry_get_promise_result (const jerry_value_t promise) /**< promise object to get the result from */
2864+
{
2865+
#if ENABLED (JERRY_ES2015_BUILTIN_PROMISE)
2866+
if (!jerry_value_is_promise (promise))
2867+
{
2868+
return jerry_throw (ecma_raise_type_error (ECMA_ERR_MSG (wrong_args_msg_p)));
2869+
}
2870+
2871+
ecma_value_t result = ecma_promise_get_result (ecma_get_object_from_value (promise));
2872+
return result;
2873+
#else /* !ENABLED (JERRY_ES2015_BUILTIN_PROMISE) */
2874+
JERRY_UNUSED (promise);
2875+
return jerry_throw (ecma_raise_type_error (ECMA_ERR_MSG ("Promise not supported.")));
2876+
#endif /* ENABLED (JERRY_ES2015_BUILTIN_PROMISE) */
2877+
} /* jerry_get_promise_result */
2878+
2879+
/**
2880+
* Get the state of a promise object.
2881+
*
2882+
* @return - the state of the promise (one of the jerry_promise_state_t enum values)
2883+
* - JERRY_PROMISE_STATE_NONE is only returned if the input is not a promise object
2884+
* or the promise support was not enabled.
2885+
*/
2886+
jerry_promise_state_t
2887+
jerry_get_promise_state (const jerry_value_t promise) /**< promise object to get the state from */
2888+
{
2889+
#if ENABLED (JERRY_ES2015_BUILTIN_PROMISE)
2890+
if (!jerry_value_is_promise (promise))
2891+
{
2892+
return JERRY_PROMISE_STATE_NONE;
2893+
}
2894+
2895+
uint8_t state = ecma_promise_get_state (ecma_get_object_from_value (promise));
2896+
2897+
JERRY_ASSERT (state < ECMA_PROMISE_STATE__COUNT);
2898+
2899+
/* Static assert above guarantees the mapping from internal type to external type. */
2900+
return (jerry_promise_state_t) (state + 1);
2901+
#else /* !ENABLED (JERRY_ES2015_BUILTIN_PROMISE) */
2902+
JERRY_UNUSED (promise);
2903+
return JERRY_PROMISE_STATE_NONE;
2904+
#endif /* ENABLED (JERRY_ES2015_BUILTIN_PROMISE) */
2905+
} /* jerry_get_promise_state */
2906+
28482907
/**
28492908
* Call the SymbolDescriptiveString ecma builtin operation on the symbol value.
28502909
*

jerry-core/ecma/operations/ecma-promise-object.c

Lines changed: 2 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -53,7 +53,7 @@ ecma_is_promise (ecma_object_t *obj_p) /**< points to object */
5353
* @return ecma value of the promise result.
5454
* Returned value must be freed with ecma_free_value
5555
*/
56-
static inline ecma_value_t
56+
ecma_value_t
5757
ecma_promise_get_result (ecma_object_t *obj_p) /**< points to promise object */
5858
{
5959
JERRY_ASSERT (ecma_is_promise (obj_p));
@@ -84,7 +84,7 @@ ecma_promise_set_result (ecma_object_t *obj_p, /**< points to promise object */
8484
*
8585
* @return the state's enum value
8686
*/
87-
static inline uint8_t JERRY_ATTR_ALWAYS_INLINE
87+
uint8_t JERRY_ATTR_ALWAYS_INLINE
8888
ecma_promise_get_state (ecma_object_t *obj_p) /**< points to promise object */
8989
{
9090
JERRY_ASSERT (ecma_is_promise (obj_p));

jerry-core/ecma/operations/ecma-promise-object.h

Lines changed: 2 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -71,6 +71,8 @@ typedef struct
7171
bool ecma_is_promise (ecma_object_t *obj_p);
7272
ecma_value_t
7373
ecma_op_create_promise_object (ecma_value_t executor, ecma_promise_executor_type_t type);
74+
uint8_t ecma_promise_get_state (ecma_object_t *promise_p);
75+
ecma_value_t ecma_promise_get_result (ecma_object_t *promise_p);
7476
ecma_value_t ecma_promise_new_capability (void);
7577
ecma_value_t
7678
ecma_promise_then (ecma_value_t promise,

jerry-core/include/jerryscript-core.h

Lines changed: 12 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -545,10 +545,21 @@ bool jerry_foreach_object_property (const jerry_value_t obj_val, jerry_object_pr
545545
void *user_data_p);
546546

547547
/**
548-
* Promise resolve/reject functions.
548+
* Promise functions.
549549
*/
550550
jerry_value_t jerry_resolve_or_reject_promise (jerry_value_t promise, jerry_value_t argument, bool is_resolve);
551551

552+
typedef enum
553+
{
554+
JERRY_PROMISE_STATE_NONE = 0u,
555+
JERRY_PROMISE_STATE_PENDING,
556+
JERRY_PROMISE_STATE_FULFILLED,
557+
JERRY_PROMISE_STATE_REJECTED,
558+
} jerry_promise_state_t;
559+
560+
jerry_value_t jerry_get_promise_result (const jerry_value_t promise);
561+
jerry_promise_state_t jerry_get_promise_state (const jerry_value_t promise);
562+
552563
/**
553564
* Symbol functions.
554565
*/

tests/unit-core/test-api-promise.c

Lines changed: 178 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,178 @@
1+
/* Copyright JS Foundation and other contributors, http://js.foundation
2+
*
3+
* Licensed under the Apache License, Version 2.0 (the "License");
4+
* you may not use this file except in compliance with the License.
5+
* You may obtain a copy of the License at
6+
*
7+
* http://www.apache.org/licenses/LICENSE-2.0
8+
*
9+
* Unless required by applicable law or agreed to in writing, software
10+
* distributed under the License is distributed on an "AS IS" BASIS
11+
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
* See the License for the specific language governing permissions and
13+
* limitations under the License.
14+
*/
15+
16+
#include "jerryscript.h"
17+
#include "jerryscript-port.h"
18+
#include "jerryscript-port-default.h"
19+
#include "test-common.h"
20+
21+
static void
22+
test_promise_resolve_success (void)
23+
{
24+
jerry_value_t my_promise = jerry_create_promise ();
25+
26+
// A created promise has an undefined promise result by default and a pending state
27+
{
28+
jerry_value_t promise_result = jerry_get_promise_result (my_promise);
29+
TEST_ASSERT (jerry_value_is_undefined (promise_result));
30+
31+
jerry_promise_state_t promise_state = jerry_get_promise_state (my_promise);
32+
TEST_ASSERT (promise_state == JERRY_PROMISE_STATE_PENDING);
33+
34+
jerry_release_value (promise_result);
35+
}
36+
37+
// A resolved promise should have the result of from the resolve call and a fulfilled state
38+
{
39+
jerry_value_t resolve_result = jerry_resolve_or_reject_promise (my_promise, jerry_create_number (3), true);
40+
41+
jerry_value_t promise_result = jerry_get_promise_result (my_promise);
42+
TEST_ASSERT (jerry_value_is_number (promise_result));
43+
TEST_ASSERT (jerry_get_number_value (promise_result) == 3.0);
44+
45+
jerry_promise_state_t promise_state = jerry_get_promise_state (my_promise);
46+
TEST_ASSERT (promise_state == JERRY_PROMISE_STATE_FULFILLED);
47+
48+
jerry_release_value (promise_result);
49+
50+
jerry_release_value (resolve_result);
51+
}
52+
53+
// Resolvind a promise again does not change the result/state
54+
{
55+
jerry_value_t resolve_result = jerry_resolve_or_reject_promise (my_promise, jerry_create_number (50), false);
56+
57+
jerry_value_t promise_result = jerry_get_promise_result (my_promise);
58+
TEST_ASSERT (jerry_value_is_number (promise_result));
59+
TEST_ASSERT (jerry_get_number_value (promise_result) == 3.0);
60+
61+
jerry_promise_state_t promise_state = jerry_get_promise_state (my_promise);
62+
TEST_ASSERT (promise_state == JERRY_PROMISE_STATE_FULFILLED);
63+
64+
jerry_release_value (promise_result);
65+
66+
jerry_release_value (resolve_result);
67+
}
68+
69+
jerry_release_value (my_promise);
70+
} /* test_promise_resolve_success */
71+
72+
static void
73+
test_promise_resolve_fail (void)
74+
{
75+
jerry_value_t my_promise = jerry_create_promise ();
76+
77+
// A created promise has an undefined promise result by default and a pending state
78+
{
79+
jerry_value_t promise_result = jerry_get_promise_result (my_promise);
80+
TEST_ASSERT (jerry_value_is_undefined (promise_result));
81+
82+
jerry_promise_state_t promise_state = jerry_get_promise_state (my_promise);
83+
TEST_ASSERT (promise_state == JERRY_PROMISE_STATE_PENDING);
84+
85+
jerry_release_value (promise_result);
86+
}
87+
88+
// A resolved promise should have the result of from the resolve call and a fulfilled state
89+
{
90+
jerry_value_t error_value = jerry_create_error (JERRY_ERROR_TYPE, (const jerry_char_t *) "resolve_fail");
91+
jerry_value_t error_obj = jerry_get_value_from_error (error_value, true);
92+
jerry_value_t resolve_result = jerry_resolve_or_reject_promise (my_promise, error_obj, false);
93+
jerry_release_value (error_obj);
94+
95+
jerry_value_t promise_result = jerry_get_promise_result (my_promise);
96+
// The error is not throw that's why it is only an error object.
97+
TEST_ASSERT (jerry_value_is_object (promise_result));
98+
TEST_ASSERT (jerry_get_error_type (promise_result) == JERRY_ERROR_TYPE);
99+
100+
jerry_promise_state_t promise_state = jerry_get_promise_state (my_promise);
101+
TEST_ASSERT (promise_state == JERRY_PROMISE_STATE_REJECTED);
102+
103+
jerry_release_value (promise_result);
104+
105+
jerry_release_value (resolve_result);
106+
}
107+
108+
// Resolvind a promise again does not change the result/state
109+
{
110+
jerry_value_t resolve_result = jerry_resolve_or_reject_promise (my_promise, jerry_create_number (50), true);
111+
112+
jerry_value_t promise_result = jerry_get_promise_result (my_promise);
113+
TEST_ASSERT (jerry_value_is_object (promise_result));
114+
TEST_ASSERT (jerry_get_error_type (promise_result) == JERRY_ERROR_TYPE);
115+
116+
jerry_promise_state_t promise_state = jerry_get_promise_state (my_promise);
117+
TEST_ASSERT (promise_state == JERRY_PROMISE_STATE_REJECTED);
118+
119+
jerry_release_value (promise_result);
120+
121+
jerry_release_value (resolve_result);
122+
}
123+
124+
jerry_release_value (my_promise);
125+
} /* test_promise_resolve_fail */
126+
127+
static void
128+
test_promise_from_js (void)
129+
{
130+
const jerry_char_t test_source[] =
131+
"(new Promise(function(rs ,rj) { rs(30); })).then(function(v) { return v + 1; })";
132+
133+
jerry_value_t parsed_code_val = jerry_parse (NULL,
134+
0,
135+
test_source,
136+
sizeof (test_source) - 1,
137+
JERRY_PARSE_NO_OPTS);
138+
TEST_ASSERT (!jerry_value_is_error (parsed_code_val));
139+
140+
jerry_value_t res = jerry_run (parsed_code_val);
141+
TEST_ASSERT (jerry_value_is_promise (res));
142+
143+
TEST_ASSERT (jerry_get_promise_state (res) == JERRY_PROMISE_STATE_PENDING);
144+
145+
jerry_value_t run_result = jerry_run_all_enqueued_jobs ();
146+
TEST_ASSERT (jerry_value_is_undefined (run_result));
147+
jerry_release_value (run_result);
148+
149+
TEST_ASSERT (jerry_get_promise_state (res) == JERRY_PROMISE_STATE_FULFILLED);
150+
jerry_value_t promise_result = jerry_get_promise_result (res);
151+
TEST_ASSERT (jerry_value_is_number (promise_result));
152+
TEST_ASSERT (jerry_get_number_value (promise_result) == 31.0);
153+
154+
jerry_release_value (promise_result);
155+
jerry_release_value (res);
156+
jerry_release_value (parsed_code_val);
157+
} /* test_promise_from_js */
158+
159+
int
160+
main (void)
161+
{
162+
if (!jerry_is_feature_enabled (JERRY_FEATURE_PROMISE))
163+
{
164+
jerry_port_log (JERRY_LOG_LEVEL_ERROR, "Promise is disabled!\n");
165+
return 0;
166+
}
167+
168+
jerry_init (JERRY_INIT_EMPTY);
169+
170+
test_promise_resolve_fail ();
171+
test_promise_resolve_success ();
172+
173+
test_promise_from_js ();
174+
175+
jerry_cleanup ();
176+
177+
return 0;
178+
} /* main */

0 commit comments

Comments
 (0)