Skip to content

Commit 7d843f1

Browse files
fizxKyle Maxwell
and
Kyle Maxwell
authored
v8.Value becomes manually releaseable (#361)
* values are manually releaseable Co-authored-by: Kyle Maxwell <[email protected]>
1 parent 1f00b50 commit 7d843f1

8 files changed

+124
-20
lines changed

CHANGELOG.md

+3
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,9 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
66

77
## [Unreleased]
88

9+
### Added
10+
- Added support for Value.release() and FunctionCallbackInfo.release(). This is useful when using v8go in a long-running context.
11+
912
### Fixed
1013
- Use string length to ensure null character-containing strings in Go/JS are not terminated early.
1114
- Object.Set with an empty key string is now supported

context.go

+6
Original file line numberDiff line numberDiff line change
@@ -81,6 +81,12 @@ func (c *Context) Isolate() *Isolate {
8181
return c.iso
8282
}
8383

84+
func (c *Context) RetainedValueCount() int {
85+
ctxMutex.Lock()
86+
defer ctxMutex.Unlock()
87+
return int(C.ContextRetainedValueCount(c.ptr))
88+
}
89+
8490
// RunScript executes the source JavaScript; origin (a.k.a. filename) provides a
8591
// reference for the script and used in the stack trace if there is an error.
8692
// error will be of type `JSError` if not nil.

context_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -104,8 +104,8 @@ func TestMemoryLeak(t *testing.T) {
104104

105105
for i := 0; i < 6000; i++ {
106106
ctx := v8.NewContext(iso)
107-
obj := ctx.Global()
108-
_ = obj.String()
107+
_ = ctx.Global()
108+
// _ = obj.String()
109109
_, _ = ctx.RunScript("2", "")
110110
ctx.Close()
111111
}

function_template.go

+7
Original file line numberDiff line numberDiff line change
@@ -37,6 +37,13 @@ func (i *FunctionCallbackInfo) Args() []*Value {
3737
return i.args
3838
}
3939

40+
func (i *FunctionCallbackInfo) Release() {
41+
for _, arg := range i.args {
42+
arg.Release()
43+
}
44+
i.this.Release()
45+
}
46+
4047
// FunctionTemplate is used to create functions at runtime.
4148
// There can only be one function created from a FunctionTemplate in a context.
4249
// The lifetime of the created function is equal to the lifetime of the context.

function_template_test.go

+39
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,45 @@ func TestFunctionTemplate_panic_on_nil_callback(t *testing.T) {
4848
defer iso.Dispose()
4949
v8.NewFunctionTemplate(iso, nil)
5050
}
51+
func TestFunctionTemplate_generates_values(t *testing.T) {
52+
t.Parallel()
53+
54+
iso := v8.NewIsolate()
55+
defer iso.Dispose()
56+
global := v8.NewObjectTemplate(iso)
57+
printfn := v8.NewFunctionTemplate(iso, func(info *v8.FunctionCallbackInfo) *v8.Value {
58+
fmt.Printf("%+v\n", info.Args())
59+
return nil
60+
})
61+
global.Set("print", printfn, v8.ReadOnly)
62+
ctx := v8.NewContext(iso, global)
63+
defer ctx.Close()
64+
ctx.RunScript("print('foo', 'bar', 0, 1)", "")
65+
if ctx.RetainedValueCount() != 6 {
66+
t.Errorf("expected 6 retained values, got: %d", ctx.RetainedValueCount())
67+
}
68+
}
69+
70+
func TestFunctionTemplate_releases_values(t *testing.T) {
71+
t.Parallel()
72+
73+
iso := v8.NewIsolate()
74+
defer iso.Dispose()
75+
global := v8.NewObjectTemplate(iso)
76+
printfn := v8.NewFunctionTemplate(iso, func(info *v8.FunctionCallbackInfo) *v8.Value {
77+
defer info.Release()
78+
fmt.Printf("%+v\n", info.Args())
79+
return nil
80+
})
81+
global.Set("print", printfn, v8.ReadOnly)
82+
ctx := v8.NewContext(iso, global)
83+
defer ctx.Close()
84+
ctx.RunScript("print('foo', 'bar', 0, 1)", "")
85+
// there is a constant factor associated with the global.
86+
if ctx.RetainedValueCount() != 1 {
87+
t.Errorf("expected 1 retained values, got: %d", ctx.RetainedValueCount())
88+
}
89+
}
5190

5291
func TestFunctionTemplateGetFunction(t *testing.T) {
5392
t.Parallel()

0 commit comments

Comments
 (0)