tc39_test.go 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490
  1. package goja
  2. import (
  3. "errors"
  4. "fmt"
  5. "gopkg.in/yaml.v2"
  6. "io/ioutil"
  7. "os"
  8. "path"
  9. "strings"
  10. "sync"
  11. "testing"
  12. )
  13. const (
  14. tc39BASE = "testdata/test262"
  15. )
  16. var (
  17. invalidFormatError = errors.New("Invalid file format")
  18. ignorableTestError = &valueSymbol{}
  19. sabStub = MustCompile("sabStub.js", `
  20. Object.defineProperty(this, "SharedArrayBuffer", {
  21. get: function() {
  22. throw IgnorableTestError;
  23. }
  24. });`,
  25. false)
  26. )
  27. var (
  28. skipList = map[string]bool{
  29. "test/language/literals/regexp/S7.8.5_A1.1_T2.js": true, // UTF-16
  30. "test/language/literals/regexp/S7.8.5_A1.4_T2.js": true, // UTF-16
  31. "test/language/literals/regexp/S7.8.5_A2.1_T2.js": true, // UTF-16
  32. "test/language/literals/regexp/S7.8.5_A2.4_T2.js": true, // UTF-16
  33. "test/built-ins/Date/prototype/toISOString/15.9.5.43-0-8.js": true, // timezone
  34. "test/built-ins/Date/prototype/toISOString/15.9.5.43-0-9.js": true, // timezone
  35. "test/built-ins/Date/prototype/toISOString/15.9.5.43-0-10.js": true, // timezone
  36. "test/annexB/built-ins/escape/escape-above-astral.js": true, // \u{xxxxx}
  37. // utf-16
  38. "test/built-ins/Array/prototype/concat/Array.prototype.concat_spreadable-string-wrapper.js": true,
  39. // class
  40. "test/language/statements/class/subclass/builtin-objects/Symbol/symbol-valid-as-extends-value.js": true,
  41. "test/language/statements/class/subclass/builtin-objects/Symbol/new-symbol-with-super-throws.js": true,
  42. "test/language/statements/class/subclass/builtin-objects/WeakSet/super-must-be-called.js": true,
  43. "test/language/statements/class/subclass/builtin-objects/WeakSet/regular-subclassing.js": true,
  44. "test/language/statements/class/subclass/builtin-objects/WeakMap/super-must-be-called.js": true,
  45. "test/language/statements/class/subclass/builtin-objects/WeakMap/regular-subclassing.js": true,
  46. "test/language/statements/class/subclass/builtin-objects/Map/super-must-be-called.js": true,
  47. "test/language/statements/class/subclass/builtin-objects/Map/regular-subclassing.js": true,
  48. "test/language/statements/class/subclass/builtin-objects/Set/super-must-be-called.js": true,
  49. "test/language/statements/class/subclass/builtin-objects/Set/regular-subclassing.js": true,
  50. "test/language/statements/class/subclass/builtin-objects/Object/replacing-prototype.js": true,
  51. "test/language/statements/class/subclass/builtin-objects/Object/regular-subclassing.js": true,
  52. "test/built-ins/Array/prototype/concat/Array.prototype.concat_non-array.js": true,
  53. "test/language/statements/class/subclass/builtin-objects/Array/length.js": true,
  54. "test/language/statements/class/subclass/builtin-objects/TypedArray/super-must-be-called.js": true,
  55. "test/language/statements/class/subclass/builtin-objects/TypedArray/regular-subclassing.js": true,
  56. "test/language/statements/class/subclass/builtin-objects/DataView/super-must-be-called.js": true,
  57. "test/language/statements/class/subclass/builtin-objects/DataView/regular-subclassing.js": true,
  58. // full unicode regexp flag
  59. "test/built-ins/RegExp/prototype/Symbol.match/u-advance-after-empty.js": true,
  60. "test/built-ins/RegExp/prototype/Symbol.match/get-unicode-error.js": true,
  61. "test/built-ins/RegExp/prototype/Symbol.match/builtin-success-u-return-val-groups.js": true,
  62. "test/built-ins/RegExp/prototype/Symbol.match/builtin-infer-unicode.js": true,
  63. // object literals
  64. "test/built-ins/Array/from/source-object-iterator-1.js": true,
  65. "test/built-ins/Array/from/source-object-iterator-2.js": true,
  66. "test/built-ins/TypedArray/prototype/fill/fill-values-conversion-once.js": true,
  67. "test/built-ins/TypedArrays/of/this-is-not-constructor.js": true,
  68. "test/built-ins/TypedArrays/of/argument-number-value-throws.js": true,
  69. "test/built-ins/TypedArrays/from/this-is-not-constructor.js": true,
  70. "test/built-ins/TypedArrays/from/set-value-abrupt-completion.js": true,
  71. "test/built-ins/TypedArrays/from/property-abrupt-completion.js": true,
  72. "test/built-ins/TypedArray/of/this-is-not-constructor.js": true,
  73. "test/built-ins/TypedArray/from/this-is-not-constructor.js": true,
  74. "test/built-ins/DataView/custom-proto-access-throws.js": true,
  75. "test/built-ins/DataView/custom-proto-access-throws-sab.js": true,
  76. // arrow-function
  77. "test/built-ins/Object/prototype/toString/proxy-function.js": true,
  78. }
  79. featuresBlackList = []string{
  80. "arrow-function",
  81. }
  82. es6WhiteList = map[string]bool{}
  83. es6IdWhiteList = []string{
  84. "8.1.2.1",
  85. "9.5",
  86. "12.9.3",
  87. "12.9.4",
  88. "19.1",
  89. "19.4",
  90. "21.1.3.14",
  91. "21.1.3.15",
  92. "21.1.3.17",
  93. "21.2.5.6",
  94. "22.1.2.1",
  95. "22.1.2.3",
  96. "22.1.2.5",
  97. "22.1.3",
  98. "22.1.4",
  99. "22.2",
  100. "23.1",
  101. "23.2",
  102. "23.3",
  103. "23.4",
  104. "24.2",
  105. "25.1.2",
  106. "26.1",
  107. "26.2",
  108. "B.2.1",
  109. "B.2.2",
  110. }
  111. esIdPrefixWhiteList = []string{
  112. "sec-array.prototype.includes",
  113. "sec-%typedarray%",
  114. }
  115. )
  116. type tc39Test struct {
  117. name string
  118. f func(t *testing.T)
  119. }
  120. type tc39TestCtx struct {
  121. base string
  122. t *testing.T
  123. prgCache map[string]*Program
  124. prgCacheLock sync.Mutex
  125. testQueue []tc39Test
  126. }
  127. type TC39MetaNegative struct {
  128. Phase, Type string
  129. }
  130. type tc39Meta struct {
  131. Negative TC39MetaNegative
  132. Includes []string
  133. Flags []string
  134. Features []string
  135. Es5id string
  136. Es6id string
  137. Esid string
  138. }
  139. func (m *tc39Meta) hasFlag(flag string) bool {
  140. for _, f := range m.Flags {
  141. if f == flag {
  142. return true
  143. }
  144. }
  145. return false
  146. }
  147. func parseTC39File(name string) (*tc39Meta, string, error) {
  148. f, err := os.Open(name)
  149. if err != nil {
  150. return nil, "", err
  151. }
  152. defer f.Close()
  153. b, err := ioutil.ReadAll(f)
  154. if err != nil {
  155. return nil, "", err
  156. }
  157. str := string(b)
  158. metaStart := strings.Index(str, "/*---")
  159. if metaStart == -1 {
  160. return nil, "", invalidFormatError
  161. } else {
  162. metaStart += 5
  163. }
  164. metaEnd := strings.Index(str, "---*/")
  165. if metaEnd == -1 || metaEnd <= metaStart {
  166. return nil, "", invalidFormatError
  167. }
  168. var meta tc39Meta
  169. err = yaml.Unmarshal([]byte(str[metaStart:metaEnd]), &meta)
  170. if err != nil {
  171. return nil, "", err
  172. }
  173. if meta.Negative.Type != "" && meta.Negative.Phase == "" {
  174. return nil, "", errors.New("negative type is set, but phase isn't")
  175. }
  176. return &meta, str, nil
  177. }
  178. func (*tc39TestCtx) detachArrayBuffer(call FunctionCall) Value {
  179. if obj, ok := call.Argument(0).(*Object); ok {
  180. if buf, ok := obj.self.(*arrayBufferObject); ok {
  181. buf.detach()
  182. return _undefined
  183. }
  184. }
  185. panic(typeError("detachArrayBuffer() is called with incompatible argument"))
  186. }
  187. func (*tc39TestCtx) throwIgnorableTestError(FunctionCall) Value {
  188. panic(ignorableTestError)
  189. }
  190. func (ctx *tc39TestCtx) runTC39Test(name, src string, meta *tc39Meta, t testing.TB) {
  191. defer func() {
  192. if x := recover(); x != nil {
  193. panic(fmt.Sprintf("panic while running %s: %v", name, x))
  194. }
  195. }()
  196. vm := New()
  197. _262 := vm.NewObject()
  198. _262.Set("detachArrayBuffer", ctx.detachArrayBuffer)
  199. _262.Set("createRealm", ctx.throwIgnorableTestError)
  200. vm.Set("$262", _262)
  201. vm.Set("IgnorableTestError", ignorableTestError)
  202. vm.RunProgram(sabStub)
  203. err, early := ctx.runTC39Script(name, src, meta.Includes, vm)
  204. if err != nil {
  205. if meta.Negative.Type == "" {
  206. if err, ok := err.(*Exception); ok {
  207. if err.Value() == ignorableTestError {
  208. t.Skip("Test threw IgnorableTestError")
  209. }
  210. }
  211. t.Fatalf("%s: %v", name, err)
  212. } else {
  213. if meta.Negative.Phase == "early" && !early || meta.Negative.Phase == "runtime" && early {
  214. t.Fatalf("%s: error %v happened at the wrong phase (expected %s)", name, err, meta.Negative.Phase)
  215. }
  216. var errType string
  217. switch err := err.(type) {
  218. case *Exception:
  219. if o, ok := err.Value().(*Object); ok {
  220. if c := o.Get("constructor"); c != nil {
  221. if c, ok := c.(*Object); ok {
  222. errType = c.Get("name").String()
  223. } else {
  224. t.Fatalf("%s: error constructor is not an object (%v)", name, o)
  225. }
  226. } else {
  227. t.Fatalf("%s: error does not have a constructor (%v)", name, o)
  228. }
  229. } else {
  230. t.Fatalf("%s: error is not an object (%v)", name, err.Value())
  231. }
  232. case *CompilerSyntaxError:
  233. errType = "SyntaxError"
  234. case *CompilerReferenceError:
  235. errType = "ReferenceError"
  236. default:
  237. t.Fatalf("%s: error is not a JS error: %v", name, err)
  238. }
  239. if errType != meta.Negative.Type {
  240. vm.vm.prg.dumpCode(t.Logf)
  241. t.Fatalf("%s: unexpected error type (%s), expected (%s)", name, errType, meta.Negative.Type)
  242. }
  243. }
  244. } else {
  245. if meta.Negative.Type != "" {
  246. vm.vm.prg.dumpCode(t.Logf)
  247. t.Fatalf("%s: Expected error: %v", name, err)
  248. }
  249. }
  250. }
  251. func (ctx *tc39TestCtx) runTC39File(name string, t testing.TB) {
  252. if skipList[name] {
  253. t.Skip("Excluded")
  254. }
  255. p := path.Join(ctx.base, name)
  256. meta, src, err := parseTC39File(p)
  257. if err != nil {
  258. //t.Fatalf("Could not parse %s: %v", name, err)
  259. t.Errorf("Could not parse %s: %v", name, err)
  260. return
  261. }
  262. if meta.Es5id == "" {
  263. skip := true
  264. //t.Logf("%s: Not ES5, skipped", name)
  265. if es6WhiteList[name] {
  266. skip = false
  267. } else {
  268. if meta.Es6id != "" {
  269. for _, prefix := range es6IdWhiteList {
  270. if strings.HasPrefix(meta.Es6id, prefix) &&
  271. (len(meta.Es6id) == len(prefix) || meta.Es6id[len(prefix)] == '.') {
  272. skip = false
  273. break
  274. }
  275. }
  276. }
  277. }
  278. if skip {
  279. if meta.Esid != "" {
  280. for _, prefix := range esIdPrefixWhiteList {
  281. if strings.HasPrefix(meta.Esid, prefix) &&
  282. (len(meta.Esid) == len(prefix) || meta.Esid[len(prefix)] == '.') {
  283. skip = false
  284. break
  285. }
  286. }
  287. }
  288. }
  289. if skip {
  290. t.Skip("Not ES5")
  291. }
  292. for _, feature := range meta.Features {
  293. for _, bl := range featuresBlackList {
  294. if feature == bl {
  295. t.Skip("Blacklisted feature")
  296. }
  297. }
  298. }
  299. }
  300. hasRaw := meta.hasFlag("raw")
  301. if hasRaw || !meta.hasFlag("onlyStrict") {
  302. //log.Printf("Running normal test: %s", name)
  303. //t.Logf("Running normal test: %s", name)
  304. ctx.runTC39Test(name, src, meta, t)
  305. }
  306. if !hasRaw && !meta.hasFlag("noStrict") {
  307. //log.Printf("Running strict test: %s", name)
  308. //t.Logf("Running strict test: %s", name)
  309. ctx.runTC39Test(name, "'use strict';\n"+src, meta, t)
  310. }
  311. }
  312. func (ctx *tc39TestCtx) init() {
  313. ctx.prgCache = make(map[string]*Program)
  314. }
  315. func (ctx *tc39TestCtx) compile(base, name string) (*Program, error) {
  316. ctx.prgCacheLock.Lock()
  317. defer ctx.prgCacheLock.Unlock()
  318. prg := ctx.prgCache[name]
  319. if prg == nil {
  320. fname := path.Join(base, name)
  321. f, err := os.Open(fname)
  322. if err != nil {
  323. return nil, err
  324. }
  325. defer f.Close()
  326. b, err := ioutil.ReadAll(f)
  327. if err != nil {
  328. return nil, err
  329. }
  330. str := string(b)
  331. prg, err = Compile(name, str, false)
  332. if err != nil {
  333. return nil, err
  334. }
  335. ctx.prgCache[name] = prg
  336. }
  337. return prg, nil
  338. }
  339. func (ctx *tc39TestCtx) runFile(base, name string, vm *Runtime) error {
  340. prg, err := ctx.compile(base, name)
  341. if err != nil {
  342. return err
  343. }
  344. _, err = vm.RunProgram(prg)
  345. return err
  346. }
  347. func (ctx *tc39TestCtx) runTC39Script(name, src string, includes []string, vm *Runtime) (err error, early bool) {
  348. early = true
  349. err = ctx.runFile(ctx.base, path.Join("harness", "assert.js"), vm)
  350. if err != nil {
  351. return
  352. }
  353. err = ctx.runFile(ctx.base, path.Join("harness", "sta.js"), vm)
  354. if err != nil {
  355. return
  356. }
  357. for _, include := range includes {
  358. err = ctx.runFile(ctx.base, path.Join("harness", include), vm)
  359. if err != nil {
  360. return
  361. }
  362. }
  363. var p *Program
  364. p, err = Compile(name, src, false)
  365. if err != nil {
  366. return
  367. }
  368. early = false
  369. _, err = vm.RunProgram(p)
  370. return
  371. }
  372. func (ctx *tc39TestCtx) runTC39Tests(name string) {
  373. files, err := ioutil.ReadDir(path.Join(ctx.base, name))
  374. if err != nil {
  375. ctx.t.Fatal(err)
  376. }
  377. for _, file := range files {
  378. if file.Name()[0] == '.' {
  379. continue
  380. }
  381. if file.IsDir() {
  382. ctx.runTC39Tests(path.Join(name, file.Name()))
  383. } else {
  384. if strings.HasSuffix(file.Name(), ".js") {
  385. name := path.Join(name, file.Name())
  386. ctx.runTest(name, func(t *testing.T) {
  387. ctx.runTC39File(name, t)
  388. })
  389. }
  390. }
  391. }
  392. }
  393. func TestTC39(t *testing.T) {
  394. if testing.Short() {
  395. t.Skip()
  396. }
  397. if _, err := os.Stat(tc39BASE); err != nil {
  398. t.Skipf("If you want to run tc39 tests, download them from https://github.com/tc39/test262 and put into %s. The last working commit is 1ba3a7c4a93fc93b3d0d7e4146f59934a896837d. (%v)", tc39BASE, err)
  399. }
  400. ctx := &tc39TestCtx{
  401. base: tc39BASE,
  402. t: t,
  403. }
  404. ctx.init()
  405. //ctx.runTC39File("test/language/types/number/8.5.1.js", t)
  406. //ctx.runTC39Tests("test/language")
  407. ctx.runTC39Tests("test/language/expressions")
  408. ctx.runTC39Tests("test/language/arguments-object")
  409. ctx.runTC39Tests("test/language/asi")
  410. ctx.runTC39Tests("test/language/directive-prologue")
  411. ctx.runTC39Tests("test/language/function-code")
  412. ctx.runTC39Tests("test/language/eval-code")
  413. ctx.runTC39Tests("test/language/global-code")
  414. ctx.runTC39Tests("test/language/identifier-resolution")
  415. ctx.runTC39Tests("test/language/identifiers")
  416. //ctx.runTC39Tests("test/language/literals") // octal sequences in strict mode
  417. ctx.runTC39Tests("test/language/punctuators")
  418. ctx.runTC39Tests("test/language/reserved-words")
  419. ctx.runTC39Tests("test/language/source-text")
  420. ctx.runTC39Tests("test/language/statements")
  421. ctx.runTC39Tests("test/language/types")
  422. ctx.runTC39Tests("test/language/white-space")
  423. ctx.runTC39Tests("test/built-ins")
  424. ctx.runTC39Tests("test/annexB/built-ins/String/prototype/substr")
  425. ctx.runTC39Tests("test/annexB/built-ins/escape")
  426. ctx.runTC39Tests("test/annexB/built-ins/unescape")
  427. ctx.flush()
  428. }