typedarrays_test.go 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544
  1. package goja
  2. import (
  3. "bytes"
  4. "testing"
  5. )
  6. func TestUint16ArrayObject(t *testing.T) {
  7. vm := New()
  8. buf := vm._newArrayBuffer(vm.global.ArrayBufferPrototype, nil)
  9. buf.data = make([]byte, 16)
  10. if nativeEndian == littleEndian {
  11. buf.data[2] = 0xFE
  12. buf.data[3] = 0xCA
  13. } else {
  14. buf.data[2] = 0xCA
  15. buf.data[3] = 0xFE
  16. }
  17. a := vm.newUint16ArrayObject(buf, 1, 1, nil)
  18. v := a.getIdx(valueInt(0), nil)
  19. if v != valueInt(0xCAFE) {
  20. t.Fatalf("v: %v", v)
  21. }
  22. }
  23. func TestArrayBufferGoWrapper(t *testing.T) {
  24. vm := New()
  25. data := []byte{0xAA, 0xBB}
  26. buf := vm.NewArrayBuffer(data)
  27. vm.Set("buf", buf)
  28. _, err := vm.RunString(`
  29. var a = new Uint8Array(buf);
  30. if (a.length !== 2 || a[0] !== 0xAA || a[1] !== 0xBB) {
  31. throw new Error(a);
  32. }
  33. `)
  34. if err != nil {
  35. t.Fatal(err)
  36. }
  37. ret, err := vm.RunString(`
  38. var b = Uint8Array.of(0xCC, 0xDD);
  39. b.buffer;
  40. `)
  41. if err != nil {
  42. t.Fatal(err)
  43. }
  44. buf1 := ret.Export().(ArrayBuffer)
  45. data1 := buf1.Bytes()
  46. if len(data1) != 2 || data1[0] != 0xCC || data1[1] != 0xDD {
  47. t.Fatal(data1)
  48. }
  49. if buf1.Detached() {
  50. t.Fatal("buf1.Detached() returned true")
  51. }
  52. if !buf1.Detach() {
  53. t.Fatal("buf1.Detach() returned false")
  54. }
  55. if !buf1.Detached() {
  56. t.Fatal("buf1.Detached() returned false")
  57. }
  58. _, err = vm.RunString(`
  59. if (b[0] !== undefined) {
  60. throw new Error("b[0] !== undefined");
  61. }
  62. `)
  63. if err != nil {
  64. t.Fatal(err)
  65. }
  66. }
  67. func TestTypedArrayIdx(t *testing.T) {
  68. const SCRIPT = `
  69. var a = new Uint8Array(1);
  70. // 32-bit integer overflow, should not panic on 32-bit architectures
  71. if (a[4294967297] !== undefined) {
  72. throw new Error("4294967297");
  73. }
  74. // Canonical non-integer
  75. a["Infinity"] = 8;
  76. if (a["Infinity"] !== undefined) {
  77. throw new Error("Infinity");
  78. }
  79. a["NaN"] = 1;
  80. if (a["NaN"] !== undefined) {
  81. throw new Error("NaN");
  82. }
  83. // Non-canonical integer
  84. a["00"] = "00";
  85. if (a["00"] !== "00") {
  86. throw new Error("00");
  87. }
  88. // Non-canonical non-integer
  89. a["1e-3"] = "1e-3";
  90. if (a["1e-3"] !== "1e-3") {
  91. throw new Error("1e-3");
  92. }
  93. if (a["0.001"] !== undefined) {
  94. throw new Error("0.001");
  95. }
  96. // Negative zero
  97. a["-0"] = 88;
  98. if (a["-0"] !== undefined) {
  99. throw new Error("-0");
  100. }
  101. if (a[0] !== 0) {
  102. throw new Error("0");
  103. }
  104. a["9007199254740992"] = 1;
  105. if (a["9007199254740992"] !== undefined) {
  106. throw new Error("9007199254740992");
  107. }
  108. a["-9007199254740992"] = 1;
  109. if (a["-9007199254740992"] !== undefined) {
  110. throw new Error("-9007199254740992");
  111. }
  112. // Safe integer overflow, not canonical (Number("9007199254740993") === 9007199254740992)
  113. a["9007199254740993"] = 1;
  114. if (a["9007199254740993"] !== 1) {
  115. throw new Error("9007199254740993");
  116. }
  117. a["-9007199254740993"] = 1;
  118. if (a["-9007199254740993"] !== 1) {
  119. throw new Error("-9007199254740993");
  120. }
  121. // Safe integer overflow, canonical Number("9007199254740994") == 9007199254740994
  122. a["9007199254740994"] = 1;
  123. if (a["9007199254740994"] !== undefined) {
  124. throw new Error("9007199254740994");
  125. }
  126. a["-9007199254740994"] = 1;
  127. if (a["-9007199254740994"] !== undefined) {
  128. throw new Error("-9007199254740994");
  129. }
  130. `
  131. testScript(SCRIPT, _undefined, t)
  132. }
  133. func TestTypedArraySetDetachedBuffer(t *testing.T) {
  134. const SCRIPT = `
  135. let sample = new Uint8Array([42]);
  136. $DETACHBUFFER(sample.buffer);
  137. sample[0] = 1;
  138. assert.sameValue(sample[0], undefined, 'sample[0] = 1 is undefined');
  139. sample['1.1'] = 1;
  140. assert.sameValue(sample['1.1'], undefined, 'sample[\'1.1\'] = 1 is undefined');
  141. sample['-0'] = 1;
  142. assert.sameValue(sample['-0'], undefined, 'sample[\'-0\'] = 1 is undefined');
  143. sample['-1'] = 1;
  144. assert.sameValue(sample['-1'], undefined, 'sample[\'-1\'] = 1 is undefined');
  145. sample['1'] = 1;
  146. assert.sameValue(sample['1'], undefined, 'sample[\'1\'] = 1 is undefined');
  147. sample['2'] = 1;
  148. assert.sameValue(sample['2'], undefined, 'sample[\'2\'] = 1 is undefined');
  149. `
  150. vm := New()
  151. vm.Set("$DETACHBUFFER", func(buf *ArrayBuffer) {
  152. buf.Detach()
  153. })
  154. vm.testScriptWithTestLib(SCRIPT, _undefined, t)
  155. }
  156. func TestTypedArrayDefinePropDetachedBuffer(t *testing.T) {
  157. const SCRIPT = `
  158. var desc = {
  159. value: 0,
  160. configurable: false,
  161. enumerable: true,
  162. writable: true
  163. };
  164. var obj = {
  165. valueOf: function() {
  166. throw new Error("valueOf() was called");
  167. }
  168. };
  169. let sample = new Uint8Array(42);
  170. $DETACHBUFFER(sample.buffer);
  171. assert.sameValue(
  172. Reflect.defineProperty(sample, "0", desc),
  173. false,
  174. 'Reflect.defineProperty(sample, "0", {value: 0, configurable: false, enumerable: true, writable: true} ) must return false'
  175. );
  176. assert.sameValue(
  177. Reflect.defineProperty(sample, "-1", desc),
  178. false,
  179. 'Reflect.defineProperty(sample, "-1", {value: 0, configurable: false, enumerable: true, writable: true} ) must return false'
  180. );
  181. assert.sameValue(
  182. Reflect.defineProperty(sample, "1.1", desc),
  183. false,
  184. 'Reflect.defineProperty(sample, "1.1", {value: 0, configurable: false, enumerable: true, writable: true} ) must return false'
  185. );
  186. assert.sameValue(
  187. Reflect.defineProperty(sample, "-0", desc),
  188. false,
  189. 'Reflect.defineProperty(sample, "-0", {value: 0, configurable: false, enumerable: true, writable: true} ) must return false'
  190. );
  191. assert.sameValue(
  192. Reflect.defineProperty(sample, "2", {
  193. configurable: true,
  194. enumerable: true,
  195. writable: true,
  196. value: obj
  197. }),
  198. false,
  199. 'Reflect.defineProperty(sample, "2", {configurable: true, enumerable: true, writable: true, value: obj}) must return false'
  200. );
  201. assert.sameValue(
  202. Reflect.defineProperty(sample, "3", {
  203. configurable: false,
  204. enumerable: false,
  205. writable: true,
  206. value: obj
  207. }),
  208. false,
  209. 'Reflect.defineProperty(sample, "3", {configurable: false, enumerable: false, writable: true, value: obj}) must return false'
  210. );
  211. assert.sameValue(
  212. Reflect.defineProperty(sample, "4", {
  213. writable: false,
  214. configurable: false,
  215. enumerable: true,
  216. value: obj
  217. }),
  218. false,
  219. 'Reflect.defineProperty("new TA(42)", "4", {writable: false, configurable: false, enumerable: true, value: obj}) must return false'
  220. );
  221. assert.sameValue(
  222. Reflect.defineProperty(sample, "42", desc),
  223. false,
  224. 'Reflect.defineProperty(sample, "42", {value: 0, configurable: false, enumerable: true, writable: true} ) must return false'
  225. );
  226. assert.sameValue(
  227. Reflect.defineProperty(sample, "43", desc),
  228. false,
  229. 'Reflect.defineProperty(sample, "43", {value: 0, configurable: false, enumerable: true, writable: true} ) must return false'
  230. );
  231. assert.sameValue(
  232. Reflect.defineProperty(sample, "5", {
  233. get: function() {}
  234. }),
  235. false,
  236. 'Reflect.defineProperty(sample, "5", {get: function() {}}) must return false'
  237. );
  238. assert.sameValue(
  239. Reflect.defineProperty(sample, "6", {
  240. configurable: false,
  241. enumerable: true,
  242. writable: true
  243. }),
  244. false,
  245. 'Reflect.defineProperty(sample, "6", {configurable: false, enumerable: true, writable: true}) must return false'
  246. );
  247. `
  248. vm := New()
  249. vm.Set("$DETACHBUFFER", func(buf *ArrayBuffer) {
  250. buf.Detach()
  251. })
  252. vm.testScriptWithTestLib(SCRIPT, _undefined, t)
  253. }
  254. func TestTypedArrayDefineProperty(t *testing.T) {
  255. const SCRIPT = `
  256. var a = new Uint8Array(1);
  257. assert.throws(TypeError, function() {
  258. Object.defineProperty(a, "1", {value: 1});
  259. });
  260. assert.sameValue(Reflect.defineProperty(a, "1", {value: 1}), false, "1");
  261. assert.throws(TypeError, function() {
  262. Object.defineProperty(a, "Infinity", {value: 8});
  263. });
  264. assert.sameValue(Reflect.defineProperty(a, "Infinity", {value: 8}), false, "Infinity");
  265. Object.defineProperty(a, "test", {value: "passed"});
  266. assert.sameValue(a.test, "passed", "string property");
  267. assert.throws(TypeError, function() {
  268. Object.defineProperty(a, "0", {value: 1, writable: false});
  269. }, "define non-writable");
  270. assert.throws(TypeError, function() {
  271. Object.defineProperty(a, "0", {get() { return 1; }});
  272. }, "define accessor");
  273. var sample = new Uint8Array([42, 42]);
  274. assert.sameValue(
  275. Reflect.defineProperty(sample, "0", {
  276. value: 8,
  277. configurable: true,
  278. enumerable: true,
  279. writable: true
  280. }),
  281. true
  282. );
  283. assert.sameValue(sample[0], 8, "property value was set");
  284. let descriptor0 = Object.getOwnPropertyDescriptor(sample, "0");
  285. assert.sameValue(descriptor0.value, 8);
  286. assert.sameValue(descriptor0.configurable, true, "configurable");
  287. assert.sameValue(descriptor0.enumerable, true);
  288. assert.sameValue(descriptor0.writable, true);
  289. `
  290. testScriptWithTestLib(SCRIPT, _undefined, t)
  291. }
  292. func TestTypedArrayGetInvalidIndex(t *testing.T) {
  293. const SCRIPT = `
  294. var TypedArray = Object.getPrototypeOf(Int8Array);
  295. var proto = TypedArray.prototype;
  296. Object.defineProperty(proto, "1", {
  297. get: function() {
  298. throw new Error("OrdinaryGet was called!");
  299. }
  300. });
  301. var a = new Uint8Array(1);
  302. assert.sameValue(a[1], undefined);
  303. assert.sameValue(a["1"], undefined);
  304. `
  305. testScriptWithTestLib(SCRIPT, _undefined, t)
  306. }
  307. func TestExportArrayBufferToBytes(t *testing.T) {
  308. vm := New()
  309. bb := []byte("test")
  310. ab := vm.NewArrayBuffer(bb)
  311. var b []byte
  312. err := vm.ExportTo(vm.ToValue(ab), &b)
  313. if err != nil {
  314. t.Fatal(err)
  315. }
  316. if !bytes.Equal(b, bb) {
  317. t.Fatal("Not equal")
  318. }
  319. err = vm.ExportTo(vm.ToValue(123), &b)
  320. if err == nil {
  321. t.Fatal("expected error")
  322. }
  323. }
  324. func TestTypedArrayExport(t *testing.T) {
  325. vm := New()
  326. t.Run("uint8", func(t *testing.T) {
  327. v, err := vm.RunString("new Uint8Array([1, 2])")
  328. if err != nil {
  329. t.Fatal(err)
  330. }
  331. if a, ok := v.Export().([]uint8); ok {
  332. if len(a) != 2 || a[0] != 1 || a[1] != 2 {
  333. t.Fatal(a)
  334. }
  335. } else {
  336. t.Fatal("Wrong export type")
  337. }
  338. _, err = vm.RunString(`{
  339. let a = new Uint8Array([1, 2]);
  340. if (a[0] !== 1 || a[1] !== 2) {
  341. throw new Error(a);
  342. }
  343. }`)
  344. if err != nil {
  345. t.Fatal(err)
  346. }
  347. })
  348. t.Run("uint8-slice", func(t *testing.T) {
  349. v, err := vm.RunString(`{
  350. const buf = new Uint8Array([1, 2]).buffer;
  351. new Uint8Array(buf, 1, 1);
  352. }`)
  353. if err != nil {
  354. t.Fatal(err)
  355. }
  356. if a, ok := v.Export().([]uint8); ok {
  357. if len(a) != 1 || a[0] != 2 {
  358. t.Fatal(a)
  359. }
  360. } else {
  361. t.Fatal("Wrong export type")
  362. }
  363. _, err = vm.RunString(`{
  364. let a = new Uint8Array([1, 2]);
  365. if (a[0] !== 1 || a[1] !== 2) {
  366. throw new Error(a);
  367. }
  368. }`)
  369. if err != nil {
  370. t.Fatal(err)
  371. }
  372. })
  373. t.Run("int8", func(t *testing.T) {
  374. v, err := vm.RunString("new Int8Array([1, -2])")
  375. if err != nil {
  376. t.Fatal(err)
  377. }
  378. if a, ok := v.Export().([]int8); ok {
  379. if len(a) != 2 || a[0] != 1 || a[1] != -2 {
  380. t.Fatal(a)
  381. }
  382. } else {
  383. t.Fatal("Wrong export type")
  384. }
  385. })
  386. t.Run("uint16", func(t *testing.T) {
  387. v, err := vm.RunString("new Uint16Array([1, 63000])")
  388. if err != nil {
  389. t.Fatal(err)
  390. }
  391. if a, ok := v.Export().([]uint16); ok {
  392. if len(a) != 2 || a[0] != 1 || a[1] != 63000 {
  393. t.Fatal(a)
  394. }
  395. } else {
  396. t.Fatal("Wrong export type")
  397. }
  398. })
  399. t.Run("int16", func(t *testing.T) {
  400. v, err := vm.RunString("new Int16Array([1, -31000])")
  401. if err != nil {
  402. t.Fatal(err)
  403. }
  404. if a, ok := v.Export().([]int16); ok {
  405. if len(a) != 2 || a[0] != 1 || a[1] != -31000 {
  406. t.Fatal(a)
  407. }
  408. } else {
  409. t.Fatal("Wrong export type")
  410. }
  411. })
  412. t.Run("uint32", func(t *testing.T) {
  413. v, err := vm.RunString("new Uint32Array([1, 123456])")
  414. if err != nil {
  415. t.Fatal(err)
  416. }
  417. if a, ok := v.Export().([]uint32); ok {
  418. if len(a) != 2 || a[0] != 1 || a[1] != 123456 {
  419. t.Fatal(a)
  420. }
  421. } else {
  422. t.Fatal("Wrong export type")
  423. }
  424. })
  425. t.Run("int32", func(t *testing.T) {
  426. v, err := vm.RunString("new Int32Array([1, -123456])")
  427. if err != nil {
  428. t.Fatal(err)
  429. }
  430. if a, ok := v.Export().([]int32); ok {
  431. if len(a) != 2 || a[0] != 1 || a[1] != -123456 {
  432. t.Fatal(a)
  433. }
  434. } else {
  435. t.Fatal("Wrong export type")
  436. }
  437. })
  438. t.Run("float32", func(t *testing.T) {
  439. v, err := vm.RunString("new Float32Array([1, -1.23456])")
  440. if err != nil {
  441. t.Fatal(err)
  442. }
  443. if a, ok := v.Export().([]float32); ok {
  444. if len(a) != 2 || a[0] != 1 || a[1] != -1.23456 {
  445. t.Fatal(a)
  446. }
  447. } else {
  448. t.Fatal("Wrong export type")
  449. }
  450. })
  451. t.Run("float64", func(t *testing.T) {
  452. v, err := vm.RunString("new Float64Array([1, -1.23456789])")
  453. if err != nil {
  454. t.Fatal(err)
  455. }
  456. if a, ok := v.Export().([]float64); ok {
  457. if len(a) != 2 || a[0] != 1 || a[1] != -1.23456789 {
  458. t.Fatal(a)
  459. }
  460. } else {
  461. t.Fatal("Wrong export type")
  462. }
  463. })
  464. t.Run("bigint64", func(t *testing.T) {
  465. v, err := vm.RunString("new BigInt64Array([18446744073709551617n, 2n])")
  466. if err != nil {
  467. t.Fatal(err)
  468. }
  469. if a, ok := v.Export().([]int64); ok {
  470. if len(a) != 2 || a[0] != 1 || a[1] != 2 {
  471. t.Fatal(a)
  472. }
  473. } else {
  474. t.Fatal("Wrong export type")
  475. }
  476. })
  477. t.Run("biguint64", func(t *testing.T) {
  478. v, err := vm.RunString("new BigUint64Array([18446744073709551617n, 2n])")
  479. if err != nil {
  480. t.Fatal(err)
  481. }
  482. if a, ok := v.Export().([]uint64); ok {
  483. if len(a) != 2 || a[0] != 1 || a[1] != 2 {
  484. t.Fatal(a)
  485. }
  486. } else {
  487. t.Fatal("Wrong export type")
  488. }
  489. })
  490. }