Add more static class property ordering tests

Since https://github.com/tc39/ecma262/pull/1490, the "length" and "name"
properties of a class are defined before any static methods. This is
tested by #2057, in test/language/computed-property-names of all places.

At the same time, static methods with "name" as the name would overwrite
the original property, but retain the original property enumeration
order. This was not previously tested. In fact, the overwriting behavior
was not tested at all for the "length" property.

This commit mends both holes in test coverage.
This commit is contained in:
Timothy Gu 2021-06-20 12:17:25 -07:00 committed by Rick Waldron
parent 162e8be99f
commit 6319cb1e2a
2 changed files with 107 additions and 12 deletions

View File

@ -0,0 +1,82 @@
// Copyright (C) 2015 the V8 project authors. All rights reserved.
// This code is governed by the BSD license found in the LICENSE file.
/*---
esid: sec-runtime-semantics-classdefinitionevaluation
description: >
Function `length` attribute not inferred in presence of static `length` method
info: |
ClassTail : ClassHeritage_opt { ClassBody_opt }
14. If constructor is empty, then [...]
b. Let F be ! CreateBuiltinFunction(steps, 0, className, « [[ConstructorKind]], [[SourceText]] », empty, constructorParent).
15. Else,
a. Let constructorInfo be ! DefineMethod of constructor with arguments proto and constructorParent.
[ This sets the length property on constructorInfo.[[Closure]]. ]
b. Let F be constructorInfo.[[Closure]].
[...]
25. For each ClassElement e of elements, do
a. If IsStatic of e is false, then [...]
b. Else,
i. Let field be ClassElementEvaluation of e with arguments F and false.
[ This overwrites the length property on F. ]
includes: [compareArray.js]
features: [generators]
---*/
class A {
static method() {
throw new Test262Error('Static method should not be executed during definition');
}
static length() {
throw new Test262Error('Static method should not be executed during definition');
}
}
assert.sameValue(typeof A.length, 'function');
assert(compareArray(Object.getOwnPropertyNames(A), ['length', 'name', 'prototype', 'method']))
var attr = 'length';
class B {
static [attr]() {
throw new Test262Error(
'Static method defined via computed property should not be executed ' +
'during definition'
);
}
}
assert.sameValue(typeof B.length, 'function');
assert(compareArray(Object.getOwnPropertyNames(B), ['length', 'name', 'prototype']))
var isDefined = false;
class C {
static get length() {
if (isDefined) {
return 'pass';
}
throw new Test262Error('Static `get` accessor should not be executed during definition');
}
}
isDefined = true;
assert.sameValue(C.length, 'pass');
assert(compareArray(Object.getOwnPropertyNames(C), ['length', 'name', 'prototype']))
class D {
static set length(_) {
throw new Test262Error('Static `set` accessor should not be executed during definition');
}
}
assert.sameValue(D.length, undefined);
assert(compareArray(Object.getOwnPropertyNames(D), ['length', 'name', 'prototype']))
class E {
static *length() {
throw new Test262Error('Static GeneratorMethod should not be executed during definition');
}
}
assert.sameValue(typeof E.length, 'function');
assert(compareArray(Object.getOwnPropertyNames(E), ['length', 'name', 'prototype']))

View File

@ -2,32 +2,41 @@
// This code is governed by the BSD license found in the LICENSE file. // This code is governed by the BSD license found in the LICENSE file.
/*--- /*---
es6id: 14.5.15 esid: sec-runtime-semantics-classdefinitionevaluation
description: > description: >
Function `name` attribute not inferred in presence of static `name` method Function `name` attribute not inferred in presence of static `name` method
info: | info: |
ClassDeclaration : class BindingIdentifier ClassTail ClassTail : ClassHeritage_opt { ClassBody_opt }
[...] 14. If constructor is empty, then [...]
4. Let hasNameProperty be HasOwnProperty(value, "name"). b. Let F be ! CreateBuiltinFunction(steps, 0, className, « [[ConstructorKind]], [[SourceText]] », empty, constructorParent).
5. ReturnIfAbrupt(hasNameProperty). 15. Else, [...]
6. If hasNameProperty is false, then perform SetFunctionName(value, d. Perform ! SetFunctionName(F, className).
className). 25. For each ClassElement e of elements, do
a. If IsStatic of e is false, then [...]
b. Else,
i. Let field be ClassElementEvaluation of e with arguments F and false.
[ This overwrites the name property on F. ]
includes: [compareArray.js]
features: [generators] features: [generators]
---*/ ---*/
class A { class A {
static method() {
throw new Test262Error('Static method should not be executed during definition');
}
static name() { static name() {
$ERROR('Static method should not be executed during definition'); throw new Test262Error('Static method should not be executed during definition');
} }
} }
assert.sameValue(typeof A.name, 'function'); assert.sameValue(typeof A.name, 'function');
assert(compareArray(Object.getOwnPropertyNames(A), ['length', 'name', 'prototype', 'method']))
var attr = 'name'; var attr = 'name';
class B { class B {
static [attr]() { static [attr]() {
$ERROR( throw new Test262Error(
'Static method defined via computed property should not be executed ' + 'Static method defined via computed property should not be executed ' +
'during definition' 'during definition'
); );
@ -35,6 +44,7 @@ class B {
} }
assert.sameValue(typeof B.name, 'function'); assert.sameValue(typeof B.name, 'function');
assert(compareArray(Object.getOwnPropertyNames(B), ['length', 'name', 'prototype']))
var isDefined = false; var isDefined = false;
class C { class C {
@ -42,25 +52,28 @@ class C {
if (isDefined) { if (isDefined) {
return 'pass'; return 'pass';
} }
$ERROR('Static `get` accessor should not be executed during definition'); throw new Test262Error('Static `get` accessor should not be executed during definition');
} }
} }
isDefined = true; isDefined = true;
assert.sameValue(C.name, 'pass'); assert.sameValue(C.name, 'pass');
assert(compareArray(Object.getOwnPropertyNames(C), ['length', 'name', 'prototype']))
class D { class D {
static set name(_) { static set name(_) {
$ERROR('Static `set` accessor should not be executed during definition'); throw new Test262Error('Static `set` accessor should not be executed during definition');
} }
} }
assert.sameValue(D.name, undefined); assert.sameValue(D.name, undefined);
assert(compareArray(Object.getOwnPropertyNames(D), ['length', 'name', 'prototype']))
class E { class E {
static *name() { static *name() {
$ERROR('Static GeneratorMethod should not be executed during definition'); throw new Test262Error('Static GeneratorMethod should not be executed during definition');
} }
} }
assert.sameValue(typeof E.name, 'function'); assert.sameValue(typeof E.name, 'function');
assert(compareArray(Object.getOwnPropertyNames(E), ['length', 'name', 'prototype']))