forked from rogchap/v8go
-
-
Notifications
You must be signed in to change notification settings - Fork 5
/
promise.go
145 lines (127 loc) · 4.13 KB
/
promise.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
// Copyright 2021 Roger Chapman and the v8go contributors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
package v8go
// #include <stdlib.h>
// #include "v8go.h"
import "C"
import (
"errors"
)
// PromiseState is the state of the Promise.
type PromiseState int
const (
Pending PromiseState = iota
Fulfilled
Rejected
)
// PromiseResolver is the resolver object for the promise.
// Most cases will create a new PromiseResolver and return
// the associated Promise from the resolver.
type PromiseResolver struct {
*Object
prom *Promise
}
// Promise is the JavaScript promise object defined in ES6
type Promise struct {
*Object
}
// NewPromiseResolver creates a new Promise resolver for the given context.
// The associated Promise will be in a Pending state.
func NewPromiseResolver(ctx *Context) (*PromiseResolver, error) {
if ctx == nil {
return nil, errors.New("v8go: Context is required")
}
rtn := C.NewPromiseResolver(ctx.ptr)
obj, err := objectResult(ctx, rtn)
if err != nil {
return nil, err
}
return &PromiseResolver{obj, nil}, nil
}
// GetPromise returns the associated Promise object for this resolver.
// The Promise object is unique to the resolver and returns the same object
// on multiple calls.
func (r *PromiseResolver) GetPromise() *Promise {
if r.prom == nil {
ptr := C.PromiseResolverGetPromise(r.ptr)
val := &Value{ptr, r.ctx}
r.prom = &Promise{&Object{val}}
}
return r.prom
}
// Resolve invokes the Promise resolve state with the given value.
// The Promise state will transition from Pending to Fulfilled.
func (r *PromiseResolver) Resolve(val Valuer) bool {
return C.PromiseResolverResolve(r.ptr, val.value().ptr) != 0
}
// Reject invokes the Promise reject state with the given value.
// The Promise state will transition from Pending to Rejected.
func (r *PromiseResolver) Reject(err *Value) bool {
return C.PromiseResolverReject(r.ptr, err.ptr) != 0
}
// State returns the current state of the Promise.
func (p *Promise) State() PromiseState {
return PromiseState(C.PromiseState(p.ptr))
}
// Result is the value result of the Promise. The Promise must
// NOT be in a Pending state, otherwise may panic. Call promise.State()
// to validate state before calling for the result.
func (p *Promise) Result() *Value {
ptr := C.PromiseResult(p.ptr)
val := &Value{ptr, p.ctx}
return val
}
// Then accepts 1 or 2 callbacks.
// The first is invoked when the promise has been fulfilled.
// The second is invoked when the promise has been rejected.
// The returned Promise resolves after the callback finishes execution.
//
// V8 only invokes the callback when processing "microtasks".
// The default MicrotaskPolicy processes them when the call depth decreases to 0.
// Call (*Context).PerformMicrotaskCheckpoint to trigger it manually.
func (p *Promise) Then(cbs ...FunctionCallback) *Promise {
cbwes := make([]FunctionCallbackWithError, len(cbs))
for i, cb := range cbs {
cb := cb
cbwes[i] = func(info *FunctionCallbackInfo) (*Value, error) {
return cb(info), nil
}
}
return p.ThenWithError(cbwes...)
}
func (p *Promise) ThenWithError(cbs ...FunctionCallbackWithError) *Promise {
var rtn C.RtnValue
switch len(cbs) {
case 1:
cbID := p.ctx.iso.registerCallback(cbs[0])
rtn = C.PromiseThen(p.ptr, C.int(cbID))
case 2:
cbID1 := p.ctx.iso.registerCallback(cbs[0])
cbID2 := p.ctx.iso.registerCallback(cbs[1])
rtn = C.PromiseThen2(p.ptr, C.int(cbID1), C.int(cbID2))
default:
panic("1 or 2 callbacks required")
}
obj, err := objectResult(p.ctx, rtn)
if err != nil {
panic(err) // TODO: Return error
}
return &Promise{obj}
}
// Catch invokes the given function if the promise is rejected.
// See Then for other details.
func (p *Promise) Catch(cb FunctionCallback) *Promise {
return p.CatchWithError(func(info *FunctionCallbackInfo) (*Value, error) {
return cb(info), nil
})
}
func (p *Promise) CatchWithError(cb FunctionCallbackWithError) *Promise {
cbID := p.ctx.iso.registerCallback(cb)
rtn := C.PromiseCatch(p.ptr, C.int(cbID))
obj, err := objectResult(p.ctx, rtn)
if err != nil {
panic(err) // TODO: Return error
}
return &Promise{obj}
}