2018-10-09 12:08:23 +00:00
|
|
|
// Copyright 2018 the V8 project authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file.
|
|
|
|
|
|
|
|
// Flags: --harmony-weak-refs
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestConstructFinalizationRegistry() {
|
|
|
|
let fg = new FinalizationRegistry(() => {});
|
|
|
|
assertEquals(fg.toString(), "[object FinalizationRegistry]");
|
2019-01-30 12:06:32 +00:00
|
|
|
assertNotSame(fg.__proto__, Object.prototype);
|
|
|
|
assertSame(fg.__proto__.__proto__, Object.prototype);
|
2018-10-09 12:08:23 +00:00
|
|
|
})();
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestFinalizationRegistryConstructorCallAsFunction() {
|
2018-10-09 12:08:23 +00:00
|
|
|
let caught = false;
|
|
|
|
let message = "";
|
|
|
|
try {
|
2020-02-25 01:19:48 +00:00
|
|
|
let f = FinalizationRegistry(() => {});
|
2018-10-09 12:08:23 +00:00
|
|
|
} catch (e) {
|
|
|
|
message = e.message;
|
|
|
|
caught = true;
|
|
|
|
} finally {
|
|
|
|
assertTrue(caught);
|
2020-02-25 01:19:48 +00:00
|
|
|
assertEquals(message, "Constructor FinalizationRegistry requires 'new'");
|
2018-10-09 12:08:23 +00:00
|
|
|
}
|
|
|
|
})();
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestConstructFinalizationRegistryCleanupNotCallable() {
|
|
|
|
let message = "FinalizationRegistry: cleanup must be callable";
|
|
|
|
assertThrows(() => { let fg = new FinalizationRegistry(); }, TypeError, message);
|
|
|
|
assertThrows(() => { let fg = new FinalizationRegistry(1); }, TypeError, message);
|
|
|
|
assertThrows(() => { let fg = new FinalizationRegistry(null); }, TypeError, message);
|
2018-11-08 08:41:53 +00:00
|
|
|
})();
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestConstructFinalizationRegistryWithCallableProxyAsCleanup() {
|
2018-11-08 08:41:53 +00:00
|
|
|
let handler = {};
|
|
|
|
let obj = () => {};
|
|
|
|
let proxy = new Proxy(obj, handler);
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(proxy);
|
2018-11-08 08:41:53 +00:00
|
|
|
})();
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestConstructFinalizationRegistryWithNonCallableProxyAsCleanup() {
|
|
|
|
let message = "FinalizationRegistry: cleanup must be callable";
|
2018-11-08 08:41:53 +00:00
|
|
|
let handler = {};
|
|
|
|
let obj = {};
|
|
|
|
let proxy = new Proxy(obj, handler);
|
2020-02-25 01:19:48 +00:00
|
|
|
assertThrows(() => { let fg = new FinalizationRegistry(proxy); }, TypeError, message);
|
2018-11-08 08:41:53 +00:00
|
|
|
})();
|
|
|
|
|
2019-01-30 12:06:32 +00:00
|
|
|
(function TestRegisterWithNonObjectTarget() {
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(() => {});
|
|
|
|
let message = "FinalizationRegistry.prototype.register: target must be an object";
|
2019-01-30 12:06:32 +00:00
|
|
|
assertThrows(() => fg.register(1, "holdings"), TypeError, message);
|
|
|
|
assertThrows(() => fg.register(false, "holdings"), TypeError, message);
|
|
|
|
assertThrows(() => fg.register("foo", "holdings"), TypeError, message);
|
|
|
|
assertThrows(() => fg.register(Symbol(), "holdings"), TypeError, message);
|
|
|
|
assertThrows(() => fg.register(null, "holdings"), TypeError, message);
|
|
|
|
assertThrows(() => fg.register(undefined, "holdings"), TypeError, message);
|
2018-10-16 12:01:14 +00:00
|
|
|
})();
|
|
|
|
|
2019-01-30 12:06:32 +00:00
|
|
|
(function TestRegisterWithProxy() {
|
2018-10-16 14:24:23 +00:00
|
|
|
let handler = {};
|
|
|
|
let obj = {};
|
|
|
|
let proxy = new Proxy(obj, handler);
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(() => {});
|
2019-01-30 12:06:32 +00:00
|
|
|
fg.register(proxy);
|
2018-10-16 14:24:23 +00:00
|
|
|
})();
|
|
|
|
|
2019-01-30 12:06:32 +00:00
|
|
|
(function TestRegisterTargetAndHoldingsSameValue() {
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(() => {});
|
2018-10-16 14:24:23 +00:00
|
|
|
let obj = {a: 1};
|
|
|
|
// SameValue(target, holdings) not ok
|
2019-01-30 12:06:32 +00:00
|
|
|
assertThrows(() => fg.register(obj, obj), TypeError,
|
2020-02-25 01:19:48 +00:00
|
|
|
"FinalizationRegistry.prototype.register: target and holdings must not be same");
|
2018-10-16 14:24:23 +00:00
|
|
|
let holdings = {a: 1};
|
2019-01-30 12:06:32 +00:00
|
|
|
fg.register(obj, holdings);
|
2018-10-16 14:24:23 +00:00
|
|
|
})();
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestRegisterWithoutFinalizationRegistry() {
|
|
|
|
assertThrows(() => FinalizationRegistry.prototype.register.call({}, {}, "holdings"), TypeError);
|
2018-10-16 12:01:14 +00:00
|
|
|
// Does not throw:
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(() => {});
|
|
|
|
FinalizationRegistry.prototype.register.call(fg, {}, "holdings");
|
2018-10-16 12:01:14 +00:00
|
|
|
})();
|
|
|
|
|
2019-01-30 12:06:32 +00:00
|
|
|
(function TestUnregisterWithNonExistentKey() {
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(() => {});
|
2019-06-11 15:01:11 +00:00
|
|
|
let success = fg.unregister({"k": "whatever"});
|
|
|
|
assertFalse(success);
|
|
|
|
})();
|
|
|
|
|
2020-02-25 01:19:48 +00:00
|
|
|
(function TestUnregisterWithNonFinalizationRegistry() {
|
|
|
|
assertThrows(() => FinalizationRegistry.prototype.unregister.call({}, {}),
|
2019-06-11 15:01:11 +00:00
|
|
|
TypeError);
|
|
|
|
})();
|
|
|
|
|
|
|
|
(function TestUnregisterWithNonObjectUnregisterToken() {
|
2020-02-25 01:19:48 +00:00
|
|
|
let fg = new FinalizationRegistry(() => {});
|
2019-06-11 15:01:11 +00:00
|
|
|
assertThrows(() => fg.unregister(1), TypeError);
|
|
|
|
assertThrows(() => fg.unregister(1n), TypeError);
|
|
|
|
assertThrows(() => fg.unregister('one'), TypeError);
|
|
|
|
assertThrows(() => fg.unregister(Symbol()), TypeError);
|
|
|
|
assertThrows(() => fg.unregister(true), TypeError);
|
|
|
|
assertThrows(() => fg.unregister(false), TypeError);
|
|
|
|
assertThrows(() => fg.unregister(undefined), TypeError);
|
|
|
|
assertThrows(() => fg.unregister(null), TypeError);
|
2018-10-23 08:30:25 +00:00
|
|
|
})();
|
2018-11-05 14:21:02 +00:00
|
|
|
|
2018-12-17 08:15:02 +00:00
|
|
|
(function TestWeakRefConstructor() {
|
|
|
|
let wr = new WeakRef({});
|
2018-11-05 14:21:02 +00:00
|
|
|
assertEquals(wr.toString(), "[object WeakRef]");
|
|
|
|
assertNotSame(wr.__proto__, Object.prototype);
|
|
|
|
|
|
|
|
let deref_desc = Object.getOwnPropertyDescriptor(wr.__proto__, "deref");
|
|
|
|
assertEquals(true, deref_desc.configurable);
|
|
|
|
assertEquals(false, deref_desc.enumerable);
|
|
|
|
assertEquals("function", typeof deref_desc.value);
|
|
|
|
})();
|
|
|
|
|
2018-12-17 08:15:02 +00:00
|
|
|
(function TestWeakRefConstructorWithNonObject() {
|
|
|
|
let message = "WeakRef: target must be an object";
|
|
|
|
assertThrows(() => new WeakRef(), TypeError, message);
|
|
|
|
assertThrows(() => new WeakRef(1), TypeError, message);
|
|
|
|
assertThrows(() => new WeakRef(false), TypeError, message);
|
|
|
|
assertThrows(() => new WeakRef("foo"), TypeError, message);
|
|
|
|
assertThrows(() => new WeakRef(Symbol()), TypeError, message);
|
|
|
|
assertThrows(() => new WeakRef(null), TypeError, message);
|
|
|
|
assertThrows(() => new WeakRef(undefined), TypeError, message);
|
2018-11-05 14:21:02 +00:00
|
|
|
})();
|
|
|
|
|
2018-12-17 08:15:02 +00:00
|
|
|
(function TestWeakRefConstructorCallAsFunction() {
|
|
|
|
let caught = false;
|
|
|
|
let message = "";
|
|
|
|
try {
|
|
|
|
let f = WeakRef({});
|
|
|
|
} catch (e) {
|
|
|
|
message = e.message;
|
|
|
|
caught = true;
|
|
|
|
} finally {
|
|
|
|
assertTrue(caught);
|
|
|
|
assertEquals(message, "Constructor WeakRef requires 'new'");
|
|
|
|
}
|
2018-11-05 14:21:02 +00:00
|
|
|
})();
|
|
|
|
|
2018-12-17 08:15:02 +00:00
|
|
|
(function TestWeakRefWithProxy() {
|
2018-11-05 14:21:02 +00:00
|
|
|
let handler = {};
|
|
|
|
let obj = {};
|
|
|
|
let proxy = new Proxy(obj, handler);
|
2018-12-17 08:15:02 +00:00
|
|
|
let wr = new WeakRef(proxy);
|
2018-11-05 14:21:02 +00:00
|
|
|
})();
|