tc39_test.go 27 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754
  1. package goja
  2. import (
  3. "errors"
  4. "fmt"
  5. "io"
  6. "os"
  7. "path"
  8. "runtime/debug"
  9. "sort"
  10. "strings"
  11. "sync"
  12. "testing"
  13. "time"
  14. "github.com/Masterminds/semver/v3"
  15. "gopkg.in/yaml.v2"
  16. )
  17. const (
  18. tc39BASE = "testdata/test262"
  19. )
  20. var (
  21. invalidFormatError = errors.New("Invalid file format")
  22. ignorableTestError = newSymbol(stringEmpty)
  23. )
  24. var (
  25. skipPrefixes prefixList
  26. skipList = map[string]bool{
  27. // out-of-date (https://github.com/tc39/test262/issues/3407)
  28. "test/language/expressions/prefix-increment/S11.4.4_A6_T3.js": true,
  29. "test/language/expressions/prefix-increment/S11.4.4_A6_T2.js": true,
  30. "test/language/expressions/prefix-increment/S11.4.4_A6_T1.js": true,
  31. "test/language/expressions/prefix-decrement/S11.4.5_A6_T3.js": true,
  32. "test/language/expressions/prefix-decrement/S11.4.5_A6_T2.js": true,
  33. "test/language/expressions/prefix-decrement/S11.4.5_A6_T1.js": true,
  34. "test/language/expressions/postfix-increment/S11.3.1_A6_T3.js": true,
  35. "test/language/expressions/postfix-increment/S11.3.1_A6_T2.js": true,
  36. "test/language/expressions/postfix-increment/S11.3.1_A6_T1.js": true,
  37. "test/language/expressions/postfix-decrement/S11.3.2_A6_T3.js": true,
  38. "test/language/expressions/postfix-decrement/S11.3.2_A6_T2.js": true,
  39. "test/language/expressions/postfix-decrement/S11.3.2_A6_T1.js": true,
  40. "test/language/expressions/compound-assignment/S11.13.2_A7.1_T4.js": true,
  41. "test/language/expressions/compound-assignment/S11.13.2_A7.1_T2.js": true,
  42. "test/language/expressions/compound-assignment/S11.13.2_A7.1_T1.js": true,
  43. "test/language/expressions/compound-assignment/S11.13.2_A7.11_T4.js": true,
  44. "test/language/expressions/compound-assignment/S11.13.2_A7.11_T2.js": true,
  45. "test/language/expressions/compound-assignment/S11.13.2_A7.11_T1.js": true,
  46. "test/language/expressions/compound-assignment/S11.13.2_A7.10_T4.js": true,
  47. "test/language/expressions/compound-assignment/S11.13.2_A7.10_T2.js": true,
  48. "test/language/expressions/compound-assignment/S11.13.2_A7.10_T1.js": true,
  49. "test/language/expressions/compound-assignment/S11.13.2_A7.9_T4.js": true,
  50. "test/language/expressions/compound-assignment/S11.13.2_A7.9_T2.js": true,
  51. "test/language/expressions/compound-assignment/S11.13.2_A7.9_T1.js": true,
  52. "test/language/expressions/compound-assignment/S11.13.2_A7.8_T4.js": true,
  53. "test/language/expressions/compound-assignment/S11.13.2_A7.8_T2.js": true,
  54. "test/language/expressions/compound-assignment/S11.13.2_A7.8_T1.js": true,
  55. "test/language/expressions/compound-assignment/S11.13.2_A7.7_T4.js": true,
  56. "test/language/expressions/compound-assignment/S11.13.2_A7.7_T2.js": true,
  57. "test/language/expressions/compound-assignment/S11.13.2_A7.7_T1.js": true,
  58. "test/language/expressions/compound-assignment/S11.13.2_A7.6_T4.js": true,
  59. "test/language/expressions/compound-assignment/S11.13.2_A7.6_T2.js": true,
  60. "test/language/expressions/compound-assignment/S11.13.2_A7.6_T1.js": true,
  61. "test/language/expressions/compound-assignment/S11.13.2_A7.5_T4.js": true,
  62. "test/language/expressions/compound-assignment/S11.13.2_A7.5_T2.js": true,
  63. "test/language/expressions/compound-assignment/S11.13.2_A7.5_T1.js": true,
  64. "test/language/expressions/compound-assignment/S11.13.2_A7.4_T4.js": true,
  65. "test/language/expressions/compound-assignment/S11.13.2_A7.4_T2.js": true,
  66. "test/language/expressions/compound-assignment/S11.13.2_A7.4_T1.js": true,
  67. "test/language/expressions/compound-assignment/S11.13.2_A7.3_T4.js": true,
  68. "test/language/expressions/compound-assignment/S11.13.2_A7.3_T2.js": true,
  69. "test/language/expressions/compound-assignment/S11.13.2_A7.3_T1.js": true,
  70. "test/language/expressions/compound-assignment/S11.13.2_A7.2_T4.js": true,
  71. "test/language/expressions/compound-assignment/S11.13.2_A7.2_T2.js": true,
  72. "test/language/expressions/compound-assignment/S11.13.2_A7.2_T1.js": true,
  73. "test/language/expressions/assignment/S11.13.1_A7_T3.js": true,
  74. // timezone
  75. "test/built-ins/Date/prototype/toISOString/15.9.5.43-0-8.js": true,
  76. "test/built-ins/Date/prototype/toISOString/15.9.5.43-0-9.js": true,
  77. "test/built-ins/Date/prototype/toISOString/15.9.5.43-0-10.js": true,
  78. // floating point date calculations
  79. "test/built-ins/Date/UTC/fp-evaluation-order.js": true,
  80. // quantifier integer limit in regexp
  81. "test/built-ins/RegExp/quantifier-integer-limit.js": true,
  82. // GetFunctionRealm
  83. "test/built-ins/Function/internals/Construct/base-ctor-revoked-proxy.js": true,
  84. // Uses deprecated __lookupGetter__/__lookupSetter__
  85. "test/language/expressions/class/elements/private-getter-is-not-a-own-property.js": true,
  86. "test/language/expressions/class/elements/private-setter-is-not-a-own-property.js": true,
  87. "test/language/statements/class/elements/private-setter-is-not-a-own-property.js": true,
  88. "test/language/statements/class/elements/private-getter-is-not-a-own-property.js": true,
  89. // restricted unicode regexp syntax
  90. "test/built-ins/RegExp/unicode_restricted_quantifiable_assertion.js": true,
  91. "test/built-ins/RegExp/unicode_restricted_octal_escape.js": true,
  92. "test/built-ins/RegExp/unicode_restricted_incomple_quantifier.js": true,
  93. "test/built-ins/RegExp/unicode_restricted_incomplete_quantifier.js": true,
  94. "test/built-ins/RegExp/unicode_restricted_identity_escape_x.js": true,
  95. "test/built-ins/RegExp/unicode_restricted_identity_escape_u.js": true,
  96. "test/built-ins/RegExp/unicode_restricted_identity_escape_c.js": true,
  97. "test/built-ins/RegExp/unicode_restricted_identity_escape_alpha.js": true,
  98. "test/built-ins/RegExp/unicode_restricted_identity_escape.js": true,
  99. "test/built-ins/RegExp/unicode_restricted_brackets.js": true,
  100. "test/built-ins/RegExp/unicode_restricted_character_class_escape.js": true,
  101. "test/annexB/built-ins/RegExp/prototype/compile/pattern-string-invalid-u.js": true,
  102. // Because goja parser works in UTF-8 it is not possible to pass strings containing invalid UTF-16 code points.
  103. // This is mitigated by escaping them as \uXXXX, however because of this the RegExp source becomes
  104. // `\uXXXX` instead of `<the actual UTF-16 code point of XXXX>`.
  105. // The resulting RegExp will work exactly the same, but it causes these two tests to fail.
  106. "test/annexB/built-ins/RegExp/RegExp-leading-escape-BMP.js": true,
  107. "test/annexB/built-ins/RegExp/RegExp-trailing-escape-BMP.js": true,
  108. "test/language/literals/regexp/S7.8.5_A1.4_T2.js": true,
  109. "test/language/literals/regexp/S7.8.5_A1.1_T2.js": true,
  110. "test/language/literals/regexp/S7.8.5_A2.1_T2.js": true,
  111. "test/language/literals/regexp/S7.8.5_A2.4_T2.js": true,
  112. // async generator
  113. "test/language/expressions/optional-chaining/member-expression.js": true,
  114. "test/language/expressions/class/elements/same-line-async-method-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  115. "test/language/expressions/class/elements/same-line-async-method-rs-static-async-generator-method-privatename-identifier.js": true,
  116. "test/language/destructuring/binding/syntax/destructuring-object-parameters-function-arguments-length.js": true,
  117. "test/language/destructuring/binding/syntax/destructuring-array-parameters-function-arguments-length.js": true,
  118. "test/language/comments/hashbang/function-constructor.js": true,
  119. "test/language/statements/class/elements/after-same-line-static-async-method-rs-static-async-generator-method-privatename-identifier.js": true,
  120. "test/language/statements/class/elements/after-same-line-static-async-method-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  121. "test/language/statements/class/elements/same-line-async-method-rs-static-async-generator-method-privatename-identifier.js": true,
  122. "test/language/statements/class/elements/same-line-async-method-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  123. "test/language/expressions/class/elements/after-same-line-static-async-method-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  124. "test/language/expressions/class/elements/after-same-line-static-async-method-rs-static-async-generator-method-privatename-identifier.js": true,
  125. "test/built-ins/Object/seal/seal-asyncgeneratorfunction.js": true,
  126. "test/language/statements/switch/scope-lex-async-generator.js": true,
  127. "test/language/statements/class/elements/private-async-generator-method-name.js": true,
  128. "test/language/expressions/class/elements/private-async-generator-method-name.js": true,
  129. "test/language/expressions/async-generator/name.js": true,
  130. "test/language/statements/class/elements/same-line-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  131. "test/language/statements/class/elements/same-line-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  132. "test/language/statements/class/elements/new-sc-line-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  133. "test/language/statements/class/elements/new-sc-line-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  134. "test/language/statements/class/elements/after-same-line-static-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  135. "test/language/statements/class/elements/after-same-line-static-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  136. "test/language/statements/class/elements/after-same-line-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  137. "test/language/statements/class/elements/after-same-line-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  138. "test/language/expressions/class/elements/same-line-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  139. "test/language/expressions/class/elements/same-line-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  140. "test/language/expressions/class/elements/new-sc-line-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  141. "test/language/expressions/class/elements/new-sc-line-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  142. "test/language/expressions/class/elements/after-same-line-static-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  143. "test/language/expressions/class/elements/after-same-line-static-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  144. "test/language/expressions/class/elements/after-same-line-gen-rs-static-async-generator-method-privatename-identifier.js": true,
  145. "test/language/expressions/class/elements/after-same-line-gen-rs-static-async-generator-method-privatename-identifier-alt.js": true,
  146. "test/built-ins/GeneratorFunction/is-a-constructor.js": true,
  147. // async iterator
  148. "test/language/expressions/optional-chaining/iteration-statement-for-await-of.js": true,
  149. // legacy number literals
  150. "test/language/literals/numeric/non-octal-decimal-integer.js": true,
  151. "test/language/literals/string/S7.8.4_A4.3_T2.js": true,
  152. "test/language/literals/string/S7.8.4_A4.3_T1.js": true,
  153. // Regexp
  154. "test/language/literals/regexp/invalid-range-negative-lookbehind.js": true,
  155. "test/language/literals/regexp/invalid-range-lookbehind.js": true,
  156. "test/language/literals/regexp/invalid-optional-negative-lookbehind.js": true,
  157. "test/language/literals/regexp/invalid-optional-lookbehind.js": true,
  158. // unicode full case folding
  159. "test/built-ins/RegExp/unicode_full_case_folding.js": true,
  160. // FIXME bugs
  161. // Left-hand side as a CoverParenthesizedExpression
  162. "test/language/expressions/assignment/fn-name-lhs-cover.js": true,
  163. // Character \ missing from character class [\c]
  164. "test/annexB/built-ins/RegExp/RegExp-invalid-control-escape-character-class.js": true,
  165. "test/annexB/built-ins/RegExp/RegExp-control-escape-russian-letter.js": true,
  166. // Skip due to regexp named groups
  167. "test/built-ins/String/prototype/replaceAll/searchValue-replacer-RegExp-call.js": true,
  168. "test/built-ins/RegExp/nullable-quantifier.js": true,
  169. "test/built-ins/RegExp/lookahead-quantifier-match-groups.js": true,
  170. }
  171. featuresBlackList = []string{
  172. "async-iteration",
  173. "Symbol.asyncIterator",
  174. "resizable-arraybuffer",
  175. "regexp-named-groups",
  176. "regexp-duplicate-named-groups",
  177. "regexp-unicode-property-escapes",
  178. "regexp-match-indices",
  179. "regexp-modifiers",
  180. "RegExp.escape",
  181. "legacy-regexp",
  182. "tail-call-optimization",
  183. "Temporal",
  184. "import-assertions",
  185. "dynamic-import",
  186. "import.meta",
  187. "Atomics",
  188. "Atomics.waitAsync",
  189. "Atomics.pause",
  190. "FinalizationRegistry",
  191. "WeakRef",
  192. "__getter__",
  193. "__setter__",
  194. "ShadowRealm",
  195. "SharedArrayBuffer",
  196. "decorators",
  197. "regexp-v-flag",
  198. "iterator-helpers",
  199. "symbols-as-weakmap-keys",
  200. "uint8array-base64",
  201. "String.prototype.toWellFormed",
  202. "explicit-resource-management",
  203. "set-methods",
  204. "promise-try",
  205. "promise-with-resolvers",
  206. "array-grouping",
  207. "Math.sumPrecise",
  208. "Float16Array",
  209. "arraybuffer-transfer",
  210. "Array.fromAsync",
  211. "String.prototype.isWellFormed",
  212. }
  213. )
  214. var goVersion *semver.Version
  215. func init() {
  216. if info, ok := debug.ReadBuildInfo(); ok {
  217. goVersion = semver.MustParse(strings.TrimPrefix(info.GoVersion, "go"))
  218. } else {
  219. panic("Could not read build info")
  220. }
  221. skip := func(prefixes ...string) {
  222. for _, prefix := range prefixes {
  223. skipPrefixes.Add(prefix)
  224. }
  225. }
  226. if goVersion.LessThan(semver.MustParse("1.21")) {
  227. skip(
  228. // Go <1.21 only supports Unicode 13
  229. "test/language/identifiers/start-unicode-14.",
  230. "test/language/identifiers/part-unicode-14.",
  231. "test/language/identifiers/start-unicode-15.",
  232. "test/language/identifiers/part-unicode-15.",
  233. )
  234. }
  235. skip(
  236. // generators and async generators (harness/hidden-constructors.js)
  237. "test/built-ins/Async",
  238. // async generators
  239. "test/language/statements/class/elements/wrapped-in-sc-rs-static-async-generator-",
  240. "test/language/statements/class/elements/same-line-method-rs-static-async-generator-",
  241. "test/language/statements/class/elements/regular-definitions-rs-static-async-generator-",
  242. "test/language/statements/class/elements/private-static-async-generator-",
  243. "test/language/statements/class/elements/new-sc-line-method-rs-static-async-generator-",
  244. "test/language/statements/class/elements/multiple-stacked-definitions-rs-static-async-generator-",
  245. "test/language/statements/class/elements/new-no-sc-line-method-rs-static-async-generator-",
  246. "test/language/statements/class/elements/multiple-definitions-rs-static-async-generator-",
  247. "test/language/statements/class/elements/after-same-line-static-method-rs-static-async-generator-",
  248. "test/language/statements/class/elements/after-same-line-method-rs-static-async-generator-",
  249. "test/language/statements/class/elements/after-same-line-static-method-rs-static-async-generator-",
  250. "test/language/expressions/class/elements/wrapped-in-sc-rs-static-async-generator-",
  251. "test/language/expressions/class/elements/same-line-method-rs-static-async-generator-",
  252. "test/language/expressions/class/elements/regular-definitions-rs-static-async-generator-",
  253. "test/language/expressions/class/elements/private-static-async-generator-",
  254. "test/language/expressions/class/elements/new-sc-line-method-rs-static-async-generator-",
  255. "test/language/expressions/class/elements/multiple-stacked-definitions-rs-static-async-generator-",
  256. "test/language/expressions/class/elements/new-no-sc-line-method-rs-static-async-generator-",
  257. "test/language/expressions/class/elements/multiple-definitions-rs-static-async-generator-",
  258. "test/language/expressions/class/elements/after-same-line-static-method-rs-static-async-generator-",
  259. "test/language/expressions/class/elements/after-same-line-method-rs-static-async-generator-",
  260. "test/language/expressions/class/elements/after-same-line-static-method-rs-static-async-generator-",
  261. "test/language/eval-code/direct/async-gen-",
  262. // restricted unicode regexp syntax
  263. "test/language/literals/regexp/u-",
  264. // legacy octal escape in strings in strict mode
  265. "test/language/literals/string/legacy-octal-",
  266. "test/language/literals/string/legacy-non-octal-",
  267. // modules
  268. "test/language/export/",
  269. "test/language/import/",
  270. "test/language/module-code/",
  271. )
  272. }
  273. type tc39Test struct {
  274. name string
  275. f func(t *testing.T)
  276. }
  277. type tc39BenchmarkItem struct {
  278. name string
  279. duration time.Duration
  280. }
  281. type tc39BenchmarkData []tc39BenchmarkItem
  282. type tc39TestCtx struct {
  283. base string
  284. t *testing.T
  285. prgCache map[string]*Program
  286. prgCacheLock sync.Mutex
  287. enableBench bool
  288. benchmark tc39BenchmarkData
  289. benchLock sync.Mutex
  290. sabStub *Program
  291. //lint:ignore U1000 Only used with race
  292. testQueue []tc39Test
  293. }
  294. type TC39MetaNegative struct {
  295. Phase, Type string
  296. }
  297. type tc39Meta struct {
  298. Negative TC39MetaNegative
  299. Includes []string
  300. Flags []string
  301. Features []string
  302. Es5id string
  303. Es6id string
  304. Esid string
  305. }
  306. type prefixList struct {
  307. prefixes map[int]map[string]struct{}
  308. }
  309. func (pl *prefixList) Add(prefix string) {
  310. l := pl.prefixes[len(prefix)]
  311. if l == nil {
  312. l = make(map[string]struct{})
  313. if pl.prefixes == nil {
  314. pl.prefixes = make(map[int]map[string]struct{})
  315. }
  316. pl.prefixes[len(prefix)] = l
  317. }
  318. l[prefix] = struct{}{}
  319. }
  320. func (pl *prefixList) Match(s string) bool {
  321. for l, prefixes := range pl.prefixes {
  322. if len(s) >= l {
  323. if _, exists := prefixes[s[:l]]; exists {
  324. return true
  325. }
  326. }
  327. }
  328. return false
  329. }
  330. func (m *tc39Meta) hasFlag(flag string) bool {
  331. for _, f := range m.Flags {
  332. if f == flag {
  333. return true
  334. }
  335. }
  336. return false
  337. }
  338. func parseTC39File(name string) (*tc39Meta, string, error) {
  339. f, err := os.Open(name)
  340. if err != nil {
  341. return nil, "", err
  342. }
  343. defer f.Close()
  344. b, err := io.ReadAll(f)
  345. if err != nil {
  346. return nil, "", err
  347. }
  348. str := string(b)
  349. metaStart := strings.Index(str, "/*---")
  350. if metaStart == -1 {
  351. return nil, "", invalidFormatError
  352. } else {
  353. metaStart += 5
  354. }
  355. metaEnd := strings.Index(str, "---*/")
  356. if metaEnd == -1 || metaEnd <= metaStart {
  357. return nil, "", invalidFormatError
  358. }
  359. var meta tc39Meta
  360. err = yaml.Unmarshal([]byte(str[metaStart:metaEnd]), &meta)
  361. if err != nil {
  362. return nil, "", err
  363. }
  364. if meta.Negative.Type != "" && meta.Negative.Phase == "" {
  365. return nil, "", errors.New("negative type is set, but phase isn't")
  366. }
  367. return &meta, str, nil
  368. }
  369. func (*tc39TestCtx) detachArrayBuffer(call FunctionCall) Value {
  370. if obj, ok := call.Argument(0).(*Object); ok {
  371. if buf, ok := obj.self.(*arrayBufferObject); ok {
  372. buf.detach()
  373. return _undefined
  374. }
  375. }
  376. panic(typeError("detachArrayBuffer() is called with incompatible argument"))
  377. }
  378. func (*tc39TestCtx) throwIgnorableTestError(FunctionCall) Value {
  379. panic(ignorableTestError)
  380. }
  381. func (ctx *tc39TestCtx) runTC39Test(name, src string, meta *tc39Meta, t testing.TB) {
  382. defer func() {
  383. if x := recover(); x != nil {
  384. panic(fmt.Sprintf("panic while running %s: %v", name, x))
  385. }
  386. }()
  387. vm := New()
  388. _262 := vm.NewObject()
  389. _262.Set("detachArrayBuffer", ctx.detachArrayBuffer)
  390. _262.Set("createRealm", ctx.throwIgnorableTestError)
  391. _262.Set("evalScript", func(call FunctionCall) Value {
  392. script := call.Argument(0).String()
  393. result, err := vm.RunString(script)
  394. if err != nil {
  395. panic(err)
  396. }
  397. return result
  398. })
  399. vm.Set("$262", _262)
  400. vm.Set("IgnorableTestError", ignorableTestError)
  401. vm.RunProgram(ctx.sabStub)
  402. var out []string
  403. async := meta.hasFlag("async")
  404. if async {
  405. err := ctx.runFile(ctx.base, path.Join("harness", "doneprintHandle.js"), vm)
  406. if err != nil {
  407. t.Fatal(err)
  408. }
  409. vm.Set("print", func(msg string) {
  410. out = append(out, msg)
  411. })
  412. } else {
  413. vm.Set("print", t.Log)
  414. }
  415. err, early := ctx.runTC39Script(name, src, meta.Includes, vm)
  416. if err != nil {
  417. if meta.Negative.Type == "" {
  418. if err, ok := err.(*Exception); ok {
  419. if err.Value() == ignorableTestError {
  420. t.Skip("Test threw IgnorableTestError")
  421. }
  422. }
  423. t.Fatalf("%s: %v", name, err)
  424. } else {
  425. if (meta.Negative.Phase == "early" || meta.Negative.Phase == "parse") && !early || meta.Negative.Phase == "runtime" && early {
  426. t.Fatalf("%s: error %v happened at the wrong phase (expected %s)", name, err, meta.Negative.Phase)
  427. }
  428. var errType string
  429. switch err := err.(type) {
  430. case *Exception:
  431. if o, ok := err.Value().(*Object); ok {
  432. if c := o.Get("constructor"); c != nil {
  433. if c, ok := c.(*Object); ok {
  434. errType = c.Get("name").String()
  435. } else {
  436. t.Fatalf("%s: error constructor is not an object (%v)", name, o)
  437. }
  438. } else {
  439. t.Fatalf("%s: error does not have a constructor (%v)", name, o)
  440. }
  441. } else {
  442. t.Fatalf("%s: error is not an object (%v)", name, err.Value())
  443. }
  444. case *CompilerSyntaxError:
  445. errType = "SyntaxError"
  446. case *CompilerReferenceError:
  447. errType = "ReferenceError"
  448. default:
  449. t.Fatalf("%s: error is not a JS error: %v", name, err)
  450. }
  451. if errType != meta.Negative.Type {
  452. vm.vm.prg.dumpCode(t.Logf)
  453. t.Fatalf("%s: unexpected error type (%s), expected (%s)", name, errType, meta.Negative.Type)
  454. }
  455. }
  456. } else {
  457. if meta.Negative.Type != "" {
  458. vm.vm.prg.dumpCode(t.Logf)
  459. t.Fatalf("%s: Expected error: %v", name, err)
  460. }
  461. }
  462. if vm.vm.sp != 0 {
  463. t.Fatalf("sp: %d", vm.vm.sp)
  464. }
  465. if l := len(vm.vm.iterStack); l > 0 {
  466. t.Fatalf("iter stack is not empty: %d", l)
  467. }
  468. if async {
  469. complete := false
  470. for _, line := range out {
  471. if strings.HasPrefix(line, "Test262:AsyncTestFailure:") {
  472. t.Fatal(line)
  473. } else if line == "Test262:AsyncTestComplete" {
  474. complete = true
  475. }
  476. }
  477. if !complete {
  478. for _, line := range out {
  479. t.Log(line)
  480. }
  481. t.Fatal("Test262:AsyncTestComplete was not printed")
  482. }
  483. }
  484. }
  485. func (ctx *tc39TestCtx) runTC39File(name string, t testing.TB) {
  486. if skipList[name] {
  487. t.Skip("Excluded")
  488. }
  489. if skipPrefixes.Match(name) {
  490. t.Skip("Excluded")
  491. }
  492. p := path.Join(ctx.base, name)
  493. meta, src, err := parseTC39File(p)
  494. if err != nil {
  495. //t.Fatalf("Could not parse %s: %v", name, err)
  496. t.Errorf("Could not parse %s: %v", name, err)
  497. return
  498. }
  499. if meta.hasFlag("module") {
  500. t.Skip("module")
  501. }
  502. if meta.Es5id == "" {
  503. for _, feature := range meta.Features {
  504. for _, bl := range featuresBlackList {
  505. if feature == bl {
  506. t.Skip("Blacklisted feature")
  507. }
  508. }
  509. }
  510. }
  511. var startTime time.Time
  512. if ctx.enableBench {
  513. startTime = time.Now()
  514. }
  515. hasRaw := meta.hasFlag("raw")
  516. if hasRaw || !meta.hasFlag("onlyStrict") {
  517. //log.Printf("Running normal test: %s", name)
  518. t.Logf("Running normal test: %s", name)
  519. ctx.runTC39Test(name, src, meta, t)
  520. }
  521. if !hasRaw && !meta.hasFlag("noStrict") {
  522. //log.Printf("Running strict test: %s", name)
  523. t.Logf("Running strict test: %s", name)
  524. ctx.runTC39Test(name, "'use strict';\n"+src, meta, t)
  525. }
  526. if ctx.enableBench {
  527. ctx.benchLock.Lock()
  528. ctx.benchmark = append(ctx.benchmark, tc39BenchmarkItem{
  529. name: name,
  530. duration: time.Since(startTime),
  531. })
  532. ctx.benchLock.Unlock()
  533. }
  534. }
  535. func (ctx *tc39TestCtx) init() {
  536. ctx.prgCache = make(map[string]*Program)
  537. ctx.sabStub = MustCompile("sabStub.js", `
  538. Object.defineProperty(this, "SharedArrayBuffer", {
  539. get: function() {
  540. throw IgnorableTestError;
  541. }
  542. });`,
  543. false)
  544. }
  545. func (ctx *tc39TestCtx) compile(base, name string) (*Program, error) {
  546. ctx.prgCacheLock.Lock()
  547. defer ctx.prgCacheLock.Unlock()
  548. prg := ctx.prgCache[name]
  549. if prg == nil {
  550. fname := path.Join(base, name)
  551. f, err := os.Open(fname)
  552. if err != nil {
  553. return nil, err
  554. }
  555. defer f.Close()
  556. b, err := io.ReadAll(f)
  557. if err != nil {
  558. return nil, err
  559. }
  560. str := string(b)
  561. prg, err = Compile(name, str, false)
  562. if err != nil {
  563. return nil, err
  564. }
  565. ctx.prgCache[name] = prg
  566. }
  567. return prg, nil
  568. }
  569. func (ctx *tc39TestCtx) runFile(base, name string, vm *Runtime) error {
  570. prg, err := ctx.compile(base, name)
  571. if err != nil {
  572. return err
  573. }
  574. _, err = vm.RunProgram(prg)
  575. return err
  576. }
  577. func (ctx *tc39TestCtx) runTC39Script(name, src string, includes []string, vm *Runtime) (err error, early bool) {
  578. early = true
  579. err = ctx.runFile(ctx.base, path.Join("harness", "assert.js"), vm)
  580. if err != nil {
  581. return
  582. }
  583. err = ctx.runFile(ctx.base, path.Join("harness", "sta.js"), vm)
  584. if err != nil {
  585. return
  586. }
  587. for _, include := range includes {
  588. err = ctx.runFile(ctx.base, path.Join("harness", include), vm)
  589. if err != nil {
  590. return
  591. }
  592. }
  593. var p *Program
  594. p, err = Compile(name, src, false)
  595. if err != nil {
  596. return
  597. }
  598. early = false
  599. _, err = vm.RunProgram(p)
  600. return
  601. }
  602. func (ctx *tc39TestCtx) runTC39Tests(name string) {
  603. files, err := os.ReadDir(path.Join(ctx.base, name))
  604. if err != nil {
  605. ctx.t.Fatal(err)
  606. }
  607. for _, file := range files {
  608. if file.Name()[0] == '.' {
  609. continue
  610. }
  611. if file.IsDir() {
  612. ctx.runTC39Tests(path.Join(name, file.Name()))
  613. } else {
  614. fileName := file.Name()
  615. if strings.HasSuffix(fileName, ".js") && !strings.HasSuffix(fileName, "_FIXTURE.js") {
  616. name := path.Join(name, fileName)
  617. ctx.runTest(name, func(t *testing.T) {
  618. ctx.runTC39File(name, t)
  619. })
  620. }
  621. }
  622. }
  623. }
  624. func TestTC39(t *testing.T) {
  625. if testing.Short() {
  626. t.Skip()
  627. }
  628. if _, err := os.Stat(tc39BASE); err != nil {
  629. t.Skipf("If you want to run tc39 tests, download them from https://github.com/tc39/test262 and put into %s. See .tc39_test262_checkout.sh for the latest working commit id. (%v)", tc39BASE, err)
  630. }
  631. ctx := &tc39TestCtx{
  632. base: tc39BASE,
  633. }
  634. ctx.init()
  635. //ctx.enableBench = true
  636. t.Run("tc39", func(t *testing.T) {
  637. ctx.t = t
  638. //ctx.runTC39File("test/language/types/number/8.5.1.js", t)
  639. ctx.runTC39Tests("test/language")
  640. ctx.runTC39Tests("test/built-ins")
  641. ctx.runTC39Tests("test/annexB/built-ins/String/prototype/substr")
  642. ctx.runTC39Tests("test/annexB/built-ins/String/prototype/trimLeft")
  643. ctx.runTC39Tests("test/annexB/built-ins/String/prototype/trimRight")
  644. ctx.runTC39Tests("test/annexB/built-ins/escape")
  645. ctx.runTC39Tests("test/annexB/built-ins/unescape")
  646. ctx.runTC39Tests("test/annexB/built-ins/RegExp")
  647. ctx.flush()
  648. })
  649. if ctx.enableBench {
  650. sort.Slice(ctx.benchmark, func(i, j int) bool {
  651. return ctx.benchmark[i].duration > ctx.benchmark[j].duration
  652. })
  653. bench := ctx.benchmark
  654. if len(bench) > 50 {
  655. bench = bench[:50]
  656. }
  657. for _, item := range bench {
  658. fmt.Printf("%s\t%d\n", item.name, item.duration/time.Millisecond)
  659. }
  660. }
  661. }