js: ast.GoExpr support using promises on JS backend (#12749)
parent
c23ebec944
commit
1cb06a2de4
|
@ -1,4 +1,4 @@
|
||||||
module promise
|
module builtin
|
||||||
|
|
||||||
pub interface JS.Promise {
|
pub interface JS.Promise {
|
||||||
then(onFullfilled JS.Any, onRejected JS.Any)
|
then(onFullfilled JS.Any, onRejected JS.Any)
|
||||||
|
@ -19,7 +19,7 @@ mut:
|
||||||
promise JS.Promise [noinit]
|
promise JS.Promise [noinit]
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new<T, E>(executor fn (resolve fn (T), reject fn (E))) Promise<T, E> {
|
pub fn promise_new<T, E>(executor fn (resolve fn (T), reject fn (E))) Promise<T, E> {
|
||||||
promise := JS.Promise.prototype.constructor(executor)
|
promise := JS.Promise.prototype.constructor(executor)
|
||||||
return Promise<T, E>{promise}
|
return Promise<T, E>{promise}
|
||||||
}
|
}
|
||||||
|
@ -40,20 +40,20 @@ pub fn (p Promise<T, E>) finally(callback fn ()) Promise<int, int> {
|
||||||
}
|
}
|
||||||
|
|
||||||
// reject<E> returns promise which was rejected because of specified error
|
// reject<E> returns promise which was rejected because of specified error
|
||||||
pub fn reject<E>(error E) Promise<int, E> {
|
pub fn promise_reject<E>(error E) Promise<int, E> {
|
||||||
promise := JS.Promise.reject(error)
|
promise := JS.Promise.reject(error)
|
||||||
return Promise<int, E>{promise}
|
return Promise<int, E>{promise}
|
||||||
}
|
}
|
||||||
|
|
||||||
// resolve<E> returns promise which was resolved with specified value
|
// resolve<E> returns promise which was resolved with specified value
|
||||||
pub fn resolve<T>(result T) Promise<T, int> {
|
pub fn promise_resolve<T>(result T) Promise<T, int> {
|
||||||
promise := JS.Promise.resolve(error)
|
promise := JS.Promise.resolve(error)
|
||||||
return Promise<T, int>{promise}
|
return Promise<T, int>{promise}
|
||||||
}
|
}
|
||||||
|
|
||||||
// race returns returns a promise that fulfills or rejects as soon as one of
|
// race returns returns a promise that fulfills or rejects as soon as one of
|
||||||
// the promises in an iterable fulfills or rejects, with the value or reason from that promise.
|
// the promises in an iterable fulfills or rejects, with the value or reason from that promise.
|
||||||
pub fn race<T, E>(promises []Promise<T, E>) Promise<T, E> {
|
pub fn promise_race<T, E>(promises []Promise<T, E>) Promise<T, E> {
|
||||||
promises_ := JS.Array.prototype.constructor()
|
promises_ := JS.Array.prototype.constructor()
|
||||||
|
|
||||||
for elem in promises {
|
for elem in promises {
|
|
@ -1,3 +0,0 @@
|
||||||
// Promise API wrapper
|
|
||||||
|
|
||||||
module promise
|
|
|
@ -1,18 +0,0 @@
|
||||||
module promise
|
|
||||||
|
|
||||||
fn test_promise() {
|
|
||||||
// TODO: For some reason compiler errors: "error: unknown function: js.promise.new", fix this
|
|
||||||
/*
|
|
||||||
p := new<int, f64>(fn (resolve_ fn (x int), reject_ fn (x f64)) {
|
|
||||||
println('Promise code')
|
|
||||||
assert true
|
|
||||||
resolve_(42)
|
|
||||||
})
|
|
||||||
p.then(fn (val int) {
|
|
||||||
println('resolved')
|
|
||||||
assert val == 42
|
|
||||||
}, fn (fail f64) {
|
|
||||||
println('rejected')
|
|
||||||
assert false
|
|
||||||
})*/
|
|
||||||
}
|
|
|
@ -830,6 +830,26 @@ pub fn (t &Table) chan_cname(elem_type Type, is_mut bool) string {
|
||||||
return 'chan_$elem_type_sym.cname' + suffix
|
return 'chan_$elem_type_sym.cname' + suffix
|
||||||
}
|
}
|
||||||
|
|
||||||
|
[inline]
|
||||||
|
pub fn (t &Table) promise_name(return_type Type) string {
|
||||||
|
if return_type.idx() == void_type_idx {
|
||||||
|
return 'Promise<JS.Any,JS.Any>'
|
||||||
|
}
|
||||||
|
|
||||||
|
return_type_sym := t.get_type_symbol(return_type)
|
||||||
|
return 'Promise<$return_type_sym.name, JS.Any>'
|
||||||
|
}
|
||||||
|
|
||||||
|
[inline]
|
||||||
|
pub fn (t &Table) promise_cname(return_type Type) string {
|
||||||
|
if return_type == void_type {
|
||||||
|
return 'Promise_Any_Any'
|
||||||
|
}
|
||||||
|
|
||||||
|
return_type_sym := t.get_type_symbol(return_type)
|
||||||
|
return 'Promise_${return_type_sym.name}_Any'
|
||||||
|
}
|
||||||
|
|
||||||
[inline]
|
[inline]
|
||||||
pub fn (t &Table) thread_name(return_type Type) string {
|
pub fn (t &Table) thread_name(return_type Type) string {
|
||||||
if return_type.idx() == void_type_idx {
|
if return_type.idx() == void_type_idx {
|
||||||
|
@ -943,6 +963,30 @@ pub fn (mut t Table) find_or_register_thread(return_type Type) int {
|
||||||
return t.register_type_symbol(thread_typ)
|
return t.register_type_symbol(thread_typ)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn (mut t Table) find_or_register_promise(return_type Type) int {
|
||||||
|
name := t.promise_name(return_type)
|
||||||
|
|
||||||
|
cname := t.promise_cname(return_type)
|
||||||
|
// existing
|
||||||
|
existing_idx := t.type_idxs[name]
|
||||||
|
if existing_idx > 0 {
|
||||||
|
return existing_idx
|
||||||
|
}
|
||||||
|
|
||||||
|
promise_type := TypeSymbol{
|
||||||
|
parent_idx: t.type_idxs['Promise']
|
||||||
|
kind: .struct_
|
||||||
|
name: name
|
||||||
|
cname: cname
|
||||||
|
info: Struct{
|
||||||
|
concrete_types: [return_type, t.type_idxs['JS.Any']]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// register
|
||||||
|
return t.register_type_symbol(promise_type)
|
||||||
|
}
|
||||||
|
|
||||||
pub fn (mut t Table) find_or_register_array(elem_type Type) int {
|
pub fn (mut t Table) find_or_register_array(elem_type Type) int {
|
||||||
name := t.array_name(elem_type)
|
name := t.array_name(elem_type)
|
||||||
// existing
|
// existing
|
||||||
|
|
|
@ -2560,7 +2560,7 @@ pub fn (mut c Checker) fn_call(mut node ast.CallExpr, mut continue_check &bool)
|
||||||
typ := c.expr(node.args[0].expr)
|
typ := c.expr(node.args[0].expr)
|
||||||
tsym := c.table.get_type_symbol(typ)
|
tsym := c.table.get_type_symbol(typ)
|
||||||
|
|
||||||
if !tsym.name.starts_with('js.promise.Promise<') {
|
if !tsym.name.starts_with('Promise<') {
|
||||||
c.error('JS.await: first argument must be a promise, got `$tsym.name`', node.pos)
|
c.error('JS.await: first argument must be a promise, got `$tsym.name`', node.pos)
|
||||||
return ast.void_type
|
return ast.void_type
|
||||||
}
|
}
|
||||||
|
@ -5040,8 +5040,13 @@ fn (mut c Checker) go_expr(mut node ast.GoExpr) ast.Type {
|
||||||
c.error('method in `go` statement cannot have non-reference mutable receiver',
|
c.error('method in `go` statement cannot have non-reference mutable receiver',
|
||||||
node.call_expr.left.position())
|
node.call_expr.left.position())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if c.pref.backend.is_js() {
|
||||||
|
return c.table.find_or_register_promise(ret_type)
|
||||||
|
} else {
|
||||||
return c.table.find_or_register_thread(ret_type)
|
return c.table.find_or_register_thread(ret_type)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn (mut c Checker) asm_stmt(mut stmt ast.AsmStmt) {
|
fn (mut c Checker) asm_stmt(mut stmt ast.AsmStmt) {
|
||||||
if stmt.is_goto {
|
if stmt.is_goto {
|
||||||
|
|
|
@ -17,7 +17,7 @@ const (
|
||||||
'if', 'implements', 'import', 'in', 'instanceof', 'interface', 'let', 'new', 'package',
|
'if', 'implements', 'import', 'in', 'instanceof', 'interface', 'let', 'new', 'package',
|
||||||
'private', 'protected', 'public', 'return', 'static', 'super', 'switch', 'this', 'throw',
|
'private', 'protected', 'public', 'return', 'static', 'super', 'switch', 'this', 'throw',
|
||||||
'try', 'typeof', 'var', 'void', 'while', 'with', 'yield', 'Number', 'String', 'Boolean',
|
'try', 'typeof', 'var', 'void', 'while', 'with', 'yield', 'Number', 'String', 'Boolean',
|
||||||
'Array', 'Map', 'document']
|
'Array', 'Map', 'document', 'Promise']
|
||||||
// used to generate type structs
|
// used to generate type structs
|
||||||
v_types = ['i8', 'i16', 'int', 'i64', 'byte', 'u16', 'u32', 'u64', 'f32', 'f64',
|
v_types = ['i8', 'i16', 'int', 'i64', 'byte', 'u16', 'u32', 'u64', 'f32', 'f64',
|
||||||
'int_literal', 'float_literal', 'bool', 'string', 'map', 'array', 'rune', 'any', 'voidptr']
|
'int_literal', 'float_literal', 'bool', 'string', 'map', 'array', 'rune', 'any', 'voidptr']
|
||||||
|
@ -1679,13 +1679,13 @@ fn (mut g JsGen) gen_for_stmt(it ast.ForStmt) {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut g JsGen) gen_go_expr(node ast.GoExpr) {
|
fn (mut g JsGen) gen_go_expr(node ast.GoExpr) {
|
||||||
g.writeln('await new Promise(function(resolve){')
|
g.writeln('new _v_Promise({promise: new Promise(function(resolve){')
|
||||||
g.inc_indent()
|
g.inc_indent()
|
||||||
g.write('resolve(')
|
g.write('resolve(')
|
||||||
g.expr(node.call_expr)
|
g.expr(node.call_expr)
|
||||||
g.write(');')
|
g.write(');')
|
||||||
g.dec_indent()
|
g.dec_indent()
|
||||||
g.writeln('});')
|
g.writeln('})});')
|
||||||
}
|
}
|
||||||
|
|
||||||
fn (mut g JsGen) gen_import_stmt(it ast.Import) {
|
fn (mut g JsGen) gen_import_stmt(it ast.Import) {
|
||||||
|
|
Loading…
Reference in New Issue