Javascript  |  337行  |  9.21 KB

// Copyright 2014 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

(function copyWithinArity() {
  assertEquals(Array.prototype.copyWithin.length, 2);
})();


(function copyWithinTargetAndStart() {
  // works with two arguemnts
  assertArrayEquals([4, 5, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, 3));
  assertArrayEquals([1, 4, 5, 4, 5], [1, 2, 3, 4, 5].copyWithin(1, 3));
  assertArrayEquals([1, 3, 4, 5, 5], [1, 2, 3, 4, 5].copyWithin(1, 2));
  assertArrayEquals([1, 2, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(2, 2));
})();


(function copyWithinTargetStartAndEnd() {
  // works with three arguments
  assertArrayEquals([1, 2, 3, 4, 5].copyWithin(0, 3, 4), [4, 2, 3, 4, 5]);
  assertArrayEquals([1, 2, 3, 4, 5].copyWithin(1, 3, 4), [1, 4, 3, 4, 5]);
  assertArrayEquals([1, 2, 3, 4, 5].copyWithin(1, 2, 4), [1, 3, 4, 4, 5]);
})();


(function copyWithinNegativeRelativeOffsets() {
  // works with negative arguments
  assertArrayEquals([4, 5, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, -2));
  assertArrayEquals([4, 2, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, -2, -1));
  assertArrayEquals([1, 3, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(-4, -3, -2));
  assertArrayEquals([1, 3, 4, 4, 5], [1, 2, 3, 4, 5].copyWithin(-4, -3, -1));
  assertArrayEquals([1, 3, 4, 5, 5], [1, 2, 3, 4, 5].copyWithin(-4, -3));
  // test with arguments equal to -this.length
  assertArrayEquals([1, 2, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(-5, 0));
})();


(function copyWithinArrayLikeValues() {
  // works with array-like values
  var args = (function () { return arguments; }(1, 2, 3));
  Array.prototype.copyWithin.call(args, -2, 0);
  assertArrayEquals([1, 1, 2], Array.prototype.slice.call(args));

  // [[Class]] does not change
  assertArrayEquals("[object Arguments]", Object.prototype.toString.call(args));
})();


(function copyWithinNullThis() {
  // throws on null/undefined values
  assertThrows(function() {
    return Array.prototype.copyWithin.call(null, 0, 3);
  }, TypeError);
})();


(function copyWithinUndefinedThis() {
  assertThrows(function() {
    return Array.prototype.copyWithin.call(undefined, 0, 3);
  }, TypeError);
})();


// TODO(caitp): indexed properties of String are read-only and setting them
//              should throw in strict mode. See bug v8:4042
// (function copyWithinStringThis() {
//   // test with this value as string
//   assertThrows(function() {
//     return Array.prototype.copyWithin.call("hello world", 0, 3);
//   }, TypeError);
// })();


(function copyWithinNumberThis() {
  // test with this value as number
  assertEquals(34, Array.prototype.copyWithin.call(34, 0, 3).valueOf());
})();


(function copyWithinSymbolThis() {
  // test with this value as number
  var sym = Symbol("test");
  assertEquals(sym, Array.prototype.copyWithin.call(sym, 0, 3).valueOf());
})();


(function copyyWithinTypedArray() {
  // test with this value as TypedArray
  var buffer = new ArrayBuffer(16);
  var int32View = new Int32Array(buffer);
  for (var i=0; i<int32View.length; i++) {
    int32View[i] = i*2;
  }
  assertArrayEquals(new Int32Array([2, 4, 6, 6]),
                    Array.prototype.copyWithin.call(int32View, 0, 1));
})();


(function copyWithinSloppyArguments() {
  // if arguments object is sloppy, copyWithin must move the arguments around
  function f(a, b, c, d, e) {
    [].copyWithin.call(arguments, 1, 3);
    return [a, b, c, d, e];
  }
  assertArrayEquals([1, 4, 5, 4, 5], f(1, 2, 3, 4, 5));
})();


(function copyWithinStartLessThanTarget() {
  // test with target > start on 2 arguments
  assertArrayEquals([1, 2, 3, 1, 2], [1, 2, 3, 4, 5].copyWithin(3, 0));

  // test with target > start on 3 arguments
  assertArrayEquals([1, 2, 3, 1, 2], [1, 2, 3, 4, 5].copyWithin(3, 0, 4));
})();


(function copyWithinArrayWithHoles() {
  // test on array with holes
  var arr = new Array(6);
  for (var i = 0; i < arr.length; i += 2) {
    arr[i] = i;
  }
  assertArrayEquals([, 4, , , 4, , ], arr.copyWithin(0, 3));
})();


(function copyWithinArrayLikeWithHoles() {
  // test on array-like object with holes
  assertArrayEquals({
    length: 6,
    1: 4,
    4: 4
  }, Array.prototype.copyWithin.call({
    length: 6,
    0: 0,
    2: 2,
    4: 4
  }, 0, 3));
})();


(function copyWithinNonIntegerRelativeOffsets() {
  // test on fractional arguments
  assertArrayEquals([4, 5, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0.2, 3.9));
})();


(function copyWithinNegativeZeroTarget() {
  // test with -0
  assertArrayEquals([4, 5, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(-0, 3));
})();


(function copyWithinTargetOutsideStart() {
  // test with arguments more than this.length
  assertArrayEquals([1, 2, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, 7));

  // test with arguments less than -this.length
  assertArrayEquals([1, 2, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(-7, 0));
})();


(function copyWithinEmptyArray() {
  // test on empty array
  assertArrayEquals([], [].copyWithin(0, 3));
})();


(function copyWithinTargetCutOff() {
  // test with target range being shorter than end - start
  assertArrayEquals([1, 2, 2, 3, 4], [1, 2, 3, 4, 5].copyWithin(2, 1, 4));
})();


(function copyWithinOverlappingRanges() {
  // test overlapping ranges
  var arr = [1, 2, 3, 4, 5];
  arr.copyWithin(2, 1, 4);
  assertArrayEquals([1, 2, 2, 2, 3], arr.copyWithin(2, 1, 4));
})();


(function copyWithinStrictDelete() {
  // check that [[Delete]] is strict (non-extensible via freeze)
  assertThrows(function() {
    return Object.freeze([1, , 3, , 4, 5]).copyWithin(2, 1, 4);
  }, TypeError);

  // check that [[Delete]] is strict (non-extensible via seal)
  assertThrows(function() {
    return Object.seal([1, , 3, , 4, 5]).copyWithin(2, 1, 4);
  }, TypeError);

  // check that [[Delete]] is strict (non-extensible via preventExtensions)
  assertThrows(function() {
    return Object.preventExtensions([1, , 3, , 4, 5]).copyWithin(2, 1, 4);
  }, TypeError);
})();


(function copyWithinStrictSet() {
  // check that [[Set]] is strict (non-extensible via freeze)
  assertThrows(function() {
    return Object.freeze([1, 2, 3, 4, 5]).copyWithin(0, 3);
  }, TypeError);

  // check that [[Set]] is strict (non-extensible via seal)
  assertThrows(function() {
    return Object.seal([, 2, 3, 4, 5]).copyWithin(0, 3);
  }, TypeError);

  // check that [[Set]] is strict (non-extensible via preventExtensions)
  assertThrows(function() {
    return Object.preventExtensions([ , 2, 3, 4, 5]).copyWithin(0, 3);
  }, TypeError);
})();


(function copyWithinSetterThrows() {
  function Boom() {}
  // test if we throw in between
  var arr = Object.defineProperty([1, 2, 3, 4, 5], 1, {
    set: function () {
      throw new Boom();
    }
  });

  assertThrows(function() {
    return arr.copyWithin(1, 3);
  }, Boom);

  assertArrayEquals([1, , 3, 4, 5], arr);
})();


(function copyWithinDefaultEnd() {
  // undefined as third argument
  assertArrayEquals(
      [4, 5, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, 3, undefined));
})();


(function copyWithinGetLengthOnce() {
  // test that this.length is called only once
  var count = 0;
  var arr = Object.defineProperty({ 0: 1, 1: 2, 2: 3, 3: 4, 4: 5 }, "length", {
    get: function () {
      count++;
      return 5;
    }
  });
  Array.prototype.copyWithin.call(arr, 1, 3);
  assertEquals(1, count);

  Array.prototype.copyWithin.call(arr, 1, 3, 4);
  assertEquals(2, count);
})();


(function copyWithinLargeArray() {
  var large = 10000;

  // test on a large array
  var arr = new Array(large);
  assertArrayEquals(arr, arr.copyWithin(45, 9000));

  var expected = new Array(large);
  // test on floating point numbers
  for (var i = 0; i < large; i++) {
    arr[i] = Math.random();
    expected[i] = arr[i];
    if (i >= 9000) {
      expected[(i - 9000) + 45] = arr[i];
    }
  }
  assertArrayEquals(expected, arr.copyWithin(45, 9000));

  // test on array of objects
  for (var i = 0; i < large; i++) {
    arr[i] = { num: Math.random() };
  } + 45
  arr.copyWithin(45, 9000);

  // test copied by reference
  for (var i = 9000; i < large; ++i) {
    assertSame(arr[(i - 9000) + 45], arr[i]);
  }

  // test array length remains same
  assertEquals(large, arr.length);
})();


(function copyWithinSuperLargeLength() {
  // 2^53 - 1 is the maximum value returned from ToLength()
  var large = Math.pow(2, 53) - 1;
  var object = { length: large };

  // Initialize last 10 entries
  for (var i = 1; i <= 10; ++i) {
    object[(large - 11) + i] = { num: i };
  }

  Array.prototype.copyWithin.call(object, 1, large - 10);

  // Test copied values
  for (var i = 1; i <= 10; ++i) {
    var old_ref = object[(large - 11) + i];
    var new_ref = object[i];
    assertSame(old_ref, new_ref);
    assertSame(new_ref.num, i);
  }

  // Assert length has not changed
  assertEquals(large, object.length);
})();


(function copyWithinNullEnd() {
  // test null on third argument is converted to +0
  assertArrayEquals([1, 2, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, 3, null));
})();


(function copyWithinElementsInObjectsPrototype() {
  // tamper the global Object prototype and test this works
  Object.prototype[2] = 1;
  assertArrayEquals([4, 5, 3, 4, 5], [1, 2, 3, 4, 5].copyWithin(0, 3));
  delete Object.prototype[2];

  Object.prototype[3] = "FAKE";
  assertArrayEquals(["FAKE", 5, 3, "FAKE", 5], [1, 2, 3, , 5].copyWithin(0, 3));
  delete Object.prototype[3];
})();