typedarrays_test.go 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348
  1. package goja
  2. import "testing"
  3. func TestUint16ArrayObject(t *testing.T) {
  4. vm := New()
  5. buf := vm._newArrayBuffer(vm.global.ArrayBufferPrototype, nil)
  6. buf.data = make([]byte, 16)
  7. if nativeEndian == littleEndian {
  8. buf.data[2] = 0xFE
  9. buf.data[3] = 0xCA
  10. } else {
  11. buf.data[2] = 0xCA
  12. buf.data[3] = 0xFE
  13. }
  14. a := vm.newUint16ArrayObject(buf, 1, 1, nil)
  15. v := a.getIdx(valueInt(0), nil)
  16. if v != valueInt(0xCAFE) {
  17. t.Fatalf("v: %v", v)
  18. }
  19. }
  20. func TestArrayBufferGoWrapper(t *testing.T) {
  21. vm := New()
  22. data := []byte{0xAA, 0xBB}
  23. buf := vm.NewArrayBuffer(data)
  24. vm.Set("buf", buf)
  25. _, err := vm.RunString(`
  26. var a = new Uint8Array(buf);
  27. if (a.length !== 2 || a[0] !== 0xAA || a[1] !== 0xBB) {
  28. throw new Error(a);
  29. }
  30. `)
  31. if err != nil {
  32. t.Fatal(err)
  33. }
  34. ret, err := vm.RunString(`
  35. var b = Uint8Array.of(0xCC, 0xDD);
  36. b.buffer;
  37. `)
  38. if err != nil {
  39. t.Fatal(err)
  40. }
  41. buf1 := ret.Export().(ArrayBuffer)
  42. data1 := buf1.Bytes()
  43. if len(data1) != 2 || data1[0] != 0xCC || data1[1] != 0xDD {
  44. t.Fatal(data1)
  45. }
  46. if buf1.Detached() {
  47. t.Fatal("buf1.Detached() returned true")
  48. }
  49. if !buf1.Detach() {
  50. t.Fatal("buf1.Detach() returned false")
  51. }
  52. if !buf1.Detached() {
  53. t.Fatal("buf1.Detached() returned false")
  54. }
  55. _, err = vm.RunString(`
  56. if (b[0] !== undefined) {
  57. throw new Error("b[0] !== undefined");
  58. }
  59. `)
  60. if err != nil {
  61. t.Fatal(err)
  62. }
  63. }
  64. func TestTypedArrayIdx(t *testing.T) {
  65. const SCRIPT = `
  66. var a = new Uint8Array(1);
  67. // 32-bit integer overflow, should not panic on 32-bit architectures
  68. if (a[4294967297] !== undefined) {
  69. throw new Error("4294967297");
  70. }
  71. // Canonical non-integer
  72. a["Infinity"] = 8;
  73. if (a["Infinity"] !== undefined) {
  74. throw new Error("Infinity");
  75. }
  76. a["NaN"] = 1;
  77. if (a["NaN"] !== undefined) {
  78. throw new Error("NaN");
  79. }
  80. // Non-canonical integer
  81. a["00"] = "00";
  82. if (a["00"] !== "00") {
  83. throw new Error("00");
  84. }
  85. // Non-canonical non-integer
  86. a["1e-3"] = "1e-3";
  87. if (a["1e-3"] !== "1e-3") {
  88. throw new Error("1e-3");
  89. }
  90. if (a["0.001"] !== undefined) {
  91. throw new Error("0.001");
  92. }
  93. // Negative zero
  94. a["-0"] = 88;
  95. if (a["-0"] !== undefined) {
  96. throw new Error("-0");
  97. }
  98. if (a[0] !== 0) {
  99. throw new Error("0");
  100. }
  101. a["9007199254740992"] = 1;
  102. if (a["9007199254740992"] !== undefined) {
  103. throw new Error("9007199254740992");
  104. }
  105. a["-9007199254740992"] = 1;
  106. if (a["-9007199254740992"] !== undefined) {
  107. throw new Error("-9007199254740992");
  108. }
  109. // Safe integer overflow, not canonical (Number("9007199254740993") === 9007199254740992)
  110. a["9007199254740993"] = 1;
  111. if (a["9007199254740993"] !== 1) {
  112. throw new Error("9007199254740993");
  113. }
  114. a["-9007199254740993"] = 1;
  115. if (a["-9007199254740993"] !== 1) {
  116. throw new Error("-9007199254740993");
  117. }
  118. // Safe integer overflow, canonical Number("9007199254740994") == 9007199254740994
  119. a["9007199254740994"] = 1;
  120. if (a["9007199254740994"] !== undefined) {
  121. throw new Error("9007199254740994");
  122. }
  123. a["-9007199254740994"] = 1;
  124. if (a["-9007199254740994"] !== undefined) {
  125. throw new Error("-9007199254740994");
  126. }
  127. `
  128. testScript1(SCRIPT, _undefined, t)
  129. }
  130. func TestTypedArraySetDetachedBuffer(t *testing.T) {
  131. const SCRIPT = `
  132. let sample = new Uint8Array([42]);
  133. $DETACHBUFFER(sample.buffer);
  134. sample[0] = 1;
  135. assert.sameValue(sample[0], undefined, 'sample[0] = 1 is undefined');
  136. sample['1.1'] = 1;
  137. assert.sameValue(sample['1.1'], undefined, 'sample[\'1.1\'] = 1 is undefined');
  138. sample['-0'] = 1;
  139. assert.sameValue(sample['-0'], undefined, 'sample[\'-0\'] = 1 is undefined');
  140. sample['-1'] = 1;
  141. assert.sameValue(sample['-1'], undefined, 'sample[\'-1\'] = 1 is undefined');
  142. sample['1'] = 1;
  143. assert.sameValue(sample['1'], undefined, 'sample[\'1\'] = 1 is undefined');
  144. sample['2'] = 1;
  145. assert.sameValue(sample['2'], undefined, 'sample[\'2\'] = 1 is undefined');
  146. `
  147. vm := New()
  148. vm.Set("$DETACHBUFFER", func(buf *ArrayBuffer) {
  149. buf.Detach()
  150. })
  151. _, err := vm.RunString(TESTLIB + SCRIPT)
  152. if err != nil {
  153. t.Fatal(err)
  154. }
  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. _, err := vm.RunString(TESTLIB + SCRIPT)
  253. if err != nil {
  254. t.Fatal(err)
  255. }
  256. }
  257. func TestTypedArrayDefineProperty(t *testing.T) {
  258. const SCRIPT = `
  259. var a = new Uint8Array(1);
  260. assert.throws(TypeError, function() {
  261. Object.defineProperty(a, "1", {value: 1});
  262. });
  263. assert.sameValue(Reflect.defineProperty(a, "1", {value: 1}), false, "1");
  264. assert.throws(TypeError, function() {
  265. Object.defineProperty(a, "Infinity", {value: 8});
  266. });
  267. assert.sameValue(Reflect.defineProperty(a, "Infinity", {value: 8}), false, "Infinity");
  268. Object.defineProperty(a, "test", {value: "passed"});
  269. assert.sameValue(a.test, "passed", "string property");
  270. assert.throws(TypeError, function() {
  271. Object.defineProperty(a, "0", {value: 1, writable: false});
  272. }, "define non-writable");
  273. assert.throws(TypeError, function() {
  274. Object.defineProperty(a, "0", {get() { return 1; }});
  275. }, "define accessor");
  276. var sample = new Uint8Array([42, 42]);
  277. assert.sameValue(
  278. Reflect.defineProperty(sample, "0", {
  279. value: 8,
  280. configurable: true,
  281. enumerable: true,
  282. writable: true
  283. }),
  284. true
  285. );
  286. assert.sameValue(sample[0], 8, "property value was set");
  287. let descriptor0 = Object.getOwnPropertyDescriptor(sample, "0");
  288. assert.sameValue(descriptor0.value, 8);
  289. assert.sameValue(descriptor0.configurable, true, "configurable");
  290. assert.sameValue(descriptor0.enumerable, true);
  291. assert.sameValue(descriptor0.writable, true);
  292. `
  293. testScript1(TESTLIB+SCRIPT, _undefined, t)
  294. }
  295. func TestTypedArrayGetInvalidIndex(t *testing.T) {
  296. const SCRIPT = `
  297. var TypedArray = Object.getPrototypeOf(Int8Array);
  298. var proto = TypedArray.prototype;
  299. Object.defineProperty(proto, "1", {
  300. get: function() {
  301. throw new Error("OrdinaryGet was called!");
  302. }
  303. });
  304. var a = new Uint8Array(1);
  305. assert.sameValue(a[1], undefined);
  306. assert.sameValue(a["1"], undefined);
  307. `
  308. testScript1(TESTLIB+SCRIPT, _undefined, t)
  309. }