// Copyright 2014 the V8 project authors. All rights reserved.
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are
// met:
//
//     * Redistributions of source code must retain the above copyright
//       notice, this list of conditions and the following disclaimer.
//     * Redistributions in binary form must reproduce the above
//       copyright notice, this list of conditions and the following
//       disclaimer in the documentation and/or other materials provided
//       with the distribution.
//     * Neither the name of Google Inc. nor the names of its
//       contributors may be used to endorse or promote products derived
//       from this software without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
assertFalse("abc".startsWith("a", Infinity));

assertEquals(1, String.prototype.startsWith.length);

var testString = "Hello World";
assertTrue(testString.startsWith(""));
assertTrue(testString.startsWith("Hello"));
assertFalse(testString.startsWith("hello"));
assertFalse(testString.startsWith("Hello World!"));
assertFalse(testString.startsWith(null));
assertFalse(testString.startsWith(undefined));
assertFalse(testString.startsWith());

assertTrue("null".startsWith(null));
assertTrue("undefined".startsWith(undefined));

var georgianUnicodeString = "\u10D0\u10D1\u10D2\u10D3\u10D4\u10D5\u10D6\u10D7";
assertTrue(georgianUnicodeString.startsWith(georgianUnicodeString));
assertTrue(georgianUnicodeString.startsWith("\u10D0\u10D1\u10D2"));
assertFalse(georgianUnicodeString.startsWith("\u10D8"));

assertThrows("String.prototype.startsWith.call(null, 'test')", TypeError);
assertThrows("String.prototype.startsWith.call(null, null)", TypeError);
assertThrows("String.prototype.startsWith.call(undefined, undefined)", TypeError);

assertThrows("String.prototype.startsWith.apply(null, ['test'])", TypeError);
assertThrows("String.prototype.startsWith.apply(null, [null])", TypeError);
assertThrows("String.prototype.startsWith.apply(undefined, [undefined])", TypeError);

var TEST_INPUT = [{
  msg: "Empty string", val: ""
}, {
  msg: "Number 1234.34", val: 1234.34
}, {
  msg: "Integer number 0", val: 0
}, {
  msg: "Negative number -1", val: -1
}, {
  msg: "Boolean true", val: true
}, {
  msg: "Boolean false", val: false
}, {
  msg: "Empty array []", val: []
}, {
  msg: "Empty object {}", val: {}
}, {
  msg: "Array of size 3", val: new Array(3)
}];

function testNonStringValues() {
  var i = 0;
  var l = TEST_INPUT.length;

  for (; i < l; i++) {
    var e = TEST_INPUT[i];
    var v = e.val;
    var s = String(v);
    assertTrue(s.startsWith(v), e.msg);
    assertTrue(String.prototype.startsWith.call(v, v), e.msg);
    assertTrue(String.prototype.startsWith.apply(v, [v]), e.msg);
  }
}
testNonStringValues();

var CustomType = function(value) {
  this.startsWith = String.prototype.startsWith;
  this.toString = function() {
    return String(value);
  }
};

function testCutomType() {
  var i = 0;
  var l = TEST_INPUT.length;

  for (; i < l; i++) {
    var e = TEST_INPUT[i];
    var v = e.val;
    var o = new CustomType(v);
    assertTrue(o.startsWith(v), e.msg);
  }
}
testCutomType();

// Test cases found in FF
assertTrue("abc".startsWith("abc"));
assertTrue("abcd".startsWith("abc"));
assertTrue("abc".startsWith("a"));
assertFalse("abc".startsWith("abcd"));
assertFalse("abc".startsWith("bcde"));
assertFalse("abc".startsWith("b"));
assertTrue("abc".startsWith("abc", 0));
assertFalse("abc".startsWith("bc", 0));
assertTrue("abc".startsWith("bc", 1));
assertFalse("abc".startsWith("c", 1));
assertFalse("abc".startsWith("abc", 1));
assertTrue("abc".startsWith("c", 2));
assertFalse("abc".startsWith("d", 2));
assertFalse("abc".startsWith("dcd", 2));
assertFalse("abc".startsWith("a", 42));
assertFalse("abc".startsWith("a", Infinity));
assertTrue("abc".startsWith("a", NaN));
assertFalse("abc".startsWith("b", NaN));
assertTrue("abc".startsWith("ab", -43));
assertTrue("abc".startsWith("ab", -Infinity));
assertFalse("abc".startsWith("bc", -42));
assertFalse("abc".startsWith("bc", -Infinity));

// Test cases taken from
// https://github.com/mathiasbynens/String.prototype.startsWith/blob/master/tests/tests.js
Object.prototype[1] = 2; // try to break `arguments[1]`

assertEquals(String.prototype.startsWith.length, 1);
assertEquals(String.prototype.propertyIsEnumerable("startsWith"), false);

assertEquals("undefined".startsWith(), true);
assertEquals("undefined".startsWith(undefined), true);
assertEquals("undefined".startsWith(null), false);
assertEquals("null".startsWith(), false);
assertEquals("null".startsWith(undefined), false);
assertEquals("null".startsWith(null), true);

assertEquals("abc".startsWith(), false);
assertEquals("abc".startsWith(""), true);
assertEquals("abc".startsWith("\0"), false);
assertEquals("abc".startsWith("a"), true);
assertEquals("abc".startsWith("b"), false);
assertEquals("abc".startsWith("ab"), true);
assertEquals("abc".startsWith("bc"), false);
assertEquals("abc".startsWith("abc"), true);
assertEquals("abc".startsWith("bcd"), false);
assertEquals("abc".startsWith("abcd"), false);
assertEquals("abc".startsWith("bcde"), false);

assertEquals("abc".startsWith("", NaN), true);
assertEquals("abc".startsWith("\0", NaN), false);
assertEquals("abc".startsWith("a", NaN), true);
assertEquals("abc".startsWith("b", NaN), false);
assertEquals("abc".startsWith("ab", NaN), true);
assertEquals("abc".startsWith("bc", NaN), false);
assertEquals("abc".startsWith("abc", NaN), true);
assertEquals("abc".startsWith("bcd", NaN), false);
assertEquals("abc".startsWith("abcd", NaN), false);
assertEquals("abc".startsWith("bcde", NaN), false);

assertEquals("abc".startsWith("", false), true);
assertEquals("abc".startsWith("\0", false), false);
assertEquals("abc".startsWith("a", false), true);
assertEquals("abc".startsWith("b", false), false);
assertEquals("abc".startsWith("ab", false), true);
assertEquals("abc".startsWith("bc", false), false);
assertEquals("abc".startsWith("abc", false), true);
assertEquals("abc".startsWith("bcd", false), false);
assertEquals("abc".startsWith("abcd", false), false);
assertEquals("abc".startsWith("bcde", false), false);

assertEquals("abc".startsWith("", undefined), true);
assertEquals("abc".startsWith("\0", undefined), false);
assertEquals("abc".startsWith("a", undefined), true);
assertEquals("abc".startsWith("b", undefined), false);
assertEquals("abc".startsWith("ab", undefined), true);
assertEquals("abc".startsWith("bc", undefined), false);
assertEquals("abc".startsWith("abc", undefined), true);
assertEquals("abc".startsWith("bcd", undefined), false);
assertEquals("abc".startsWith("abcd", undefined), false);
assertEquals("abc".startsWith("bcde", undefined), false);

assertEquals("abc".startsWith("", null), true);
assertEquals("abc".startsWith("\0", null), false);
assertEquals("abc".startsWith("a", null), true);
assertEquals("abc".startsWith("b", null), false);
assertEquals("abc".startsWith("ab", null), true);
assertEquals("abc".startsWith("bc", null), false);
assertEquals("abc".startsWith("abc", null), true);
assertEquals("abc".startsWith("bcd", null), false);
assertEquals("abc".startsWith("abcd", null), false);
assertEquals("abc".startsWith("bcde", null), false);

assertEquals("abc".startsWith("", -Infinity), true);
assertEquals("abc".startsWith("\0", -Infinity), false);
assertEquals("abc".startsWith("a", -Infinity), true);
assertEquals("abc".startsWith("b", -Infinity), false);
assertEquals("abc".startsWith("ab", -Infinity), true);
assertEquals("abc".startsWith("bc", -Infinity), false);
assertEquals("abc".startsWith("abc", -Infinity), true);
assertEquals("abc".startsWith("bcd", -Infinity), false);
assertEquals("abc".startsWith("abcd", -Infinity), false);
assertEquals("abc".startsWith("bcde", -Infinity), false);

assertEquals("abc".startsWith("", -1), true);
assertEquals("abc".startsWith("\0", -1), false);
assertEquals("abc".startsWith("a", -1), true);
assertEquals("abc".startsWith("b", -1), false);
assertEquals("abc".startsWith("ab", -1), true);
assertEquals("abc".startsWith("bc", -1), false);
assertEquals("abc".startsWith("abc", -1), true);
assertEquals("abc".startsWith("bcd", -1), false);
assertEquals("abc".startsWith("abcd", -1), false);
assertEquals("abc".startsWith("bcde", -1), false);

assertEquals("abc".startsWith("", -0), true);
assertEquals("abc".startsWith("\0", -0), false);
assertEquals("abc".startsWith("a", -0), true);
assertEquals("abc".startsWith("b", -0), false);
assertEquals("abc".startsWith("ab", -0), true);
assertEquals("abc".startsWith("bc", -0), false);
assertEquals("abc".startsWith("abc", -0), true);
assertEquals("abc".startsWith("bcd", -0), false);
assertEquals("abc".startsWith("abcd", -0), false);
assertEquals("abc".startsWith("bcde", -0), false);

assertEquals("abc".startsWith("", +0), true);
assertEquals("abc".startsWith("\0", +0), false);
assertEquals("abc".startsWith("a", +0), true);
assertEquals("abc".startsWith("b", +0), false);
assertEquals("abc".startsWith("ab", +0), true);
assertEquals("abc".startsWith("bc", +0), false);
assertEquals("abc".startsWith("abc", +0), true);
assertEquals("abc".startsWith("bcd", +0), false);
assertEquals("abc".startsWith("abcd", +0), false);
assertEquals("abc".startsWith("bcde", +0), false);

assertEquals("abc".startsWith("", 1), true);
assertEquals("abc".startsWith("\0", 1), false);
assertEquals("abc".startsWith("a", 1), false);
assertEquals("abc".startsWith("b", 1), true);
assertEquals("abc".startsWith("ab", 1), false);
assertEquals("abc".startsWith("bc", 1), true);
assertEquals("abc".startsWith("abc", 1), false);
assertEquals("abc".startsWith("bcd", 1), false);
assertEquals("abc".startsWith("abcd", 1), false);
assertEquals("abc".startsWith("bcde", 1), false);

assertEquals("abc".startsWith("", +Infinity), true);
assertEquals("abc".startsWith("\0", +Infinity), false);
assertEquals("abc".startsWith("a", +Infinity), false);
assertEquals("abc".startsWith("b", +Infinity), false);
assertEquals("abc".startsWith("ab", +Infinity), false);
assertEquals("abc".startsWith("bc", +Infinity), false);
assertEquals("abc".startsWith("abc", +Infinity), false);
assertEquals("abc".startsWith("bcd", +Infinity), false);
assertEquals("abc".startsWith("abcd", +Infinity), false);
assertEquals("abc".startsWith("bcde", +Infinity), false);

assertEquals("abc".startsWith("", true), true);
assertEquals("abc".startsWith("\0", true), false);
assertEquals("abc".startsWith("a", true), false);
assertEquals("abc".startsWith("b", true), true);
assertEquals("abc".startsWith("ab", true), false);
assertEquals("abc".startsWith("bc", true), true);
assertEquals("abc".startsWith("abc", true), false);
assertEquals("abc".startsWith("bcd", true), false);
assertEquals("abc".startsWith("abcd", true), false);
assertEquals("abc".startsWith("bcde", true), false);

assertEquals("abc".startsWith("", "x"), true);
assertEquals("abc".startsWith("\0", "x"), false);
assertEquals("abc".startsWith("a", "x"), true);
assertEquals("abc".startsWith("b", "x"), false);
assertEquals("abc".startsWith("ab", "x"), true);
assertEquals("abc".startsWith("bc", "x"), false);
assertEquals("abc".startsWith("abc", "x"), true);
assertEquals("abc".startsWith("bcd", "x"), false);
assertEquals("abc".startsWith("abcd", "x"), false);
assertEquals("abc".startsWith("bcde", "x"), false);

assertEquals("[a-z]+(bar)?".startsWith("[a-z]+"), true);
assertThrows(function() { "[a-z]+(bar)?".startsWith(/[a-z]+/); }, TypeError);
assertEquals("[a-z]+(bar)?".startsWith("(bar)?", 6), true);
assertThrows(function() { "[a-z]+(bar)?".startsWith(/(bar)?/); }, TypeError);
assertThrows(function() { "[a-z]+/(bar)?/".startsWith(/(bar)?/); }, TypeError);

// http://mathiasbynens.be/notes/javascript-unicode#poo-test
var string = "I\xF1t\xEBrn\xE2ti\xF4n\xE0liz\xE6ti\xF8n\u2603\uD83D\uDCA9";
assertEquals(string.startsWith(""), true);
assertEquals(string.startsWith("\xF1t\xEBr"), false);
assertEquals(string.startsWith("\xF1t\xEBr", 1), true);
assertEquals(string.startsWith("\xE0liz\xE6"), false);
assertEquals(string.startsWith("\xE0liz\xE6", 11), true);
assertEquals(string.startsWith("\xF8n\u2603\uD83D\uDCA9"), false);
assertEquals(string.startsWith("\xF8n\u2603\uD83D\uDCA9", 18), true);
assertEquals(string.startsWith("\u2603"), false);
assertEquals(string.startsWith("\u2603", 20), true);
assertEquals(string.startsWith("\uD83D\uDCA9"), false);
assertEquals(string.startsWith("\uD83D\uDCA9", 21), true);

assertThrows(function() {
  String.prototype.startsWith.call(undefined);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.call(undefined, "b");
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.call(undefined, "b", 4);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.call(null);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.call(null, "b");
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.call(null, "b", 4);
}, TypeError);
assertEquals(String.prototype.startsWith.call(42, "2"), false);
assertEquals(String.prototype.startsWith.call(42, "4"), true);
assertEquals(String.prototype.startsWith.call(42, "b", 4), false);
assertEquals(String.prototype.startsWith.call(42, "2", 1), true);
assertEquals(String.prototype.startsWith.call(42, "2", 4), false);
assertEquals(String.prototype.startsWith.call({
  "toString": function() { return "abc"; }
}, "b", 0), false);
assertEquals(String.prototype.startsWith.call({
  "toString": function() { return "abc"; }
}, "b", 1), true);
assertEquals(String.prototype.startsWith.call({
  "toString": function() { return "abc"; }
}, "b", 2), false);
assertThrows(function() {
  String.prototype.startsWith.call({
    "toString": function() { throw RangeError(); }
  }, /./);
}, RangeError);
assertThrows(function() {
  String.prototype.startsWith.call({
    "toString": function() { return "abc"; }
  }, /./);
}, TypeError);

assertThrows(function() {
  String.prototype.startsWith.apply(undefined);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.apply(undefined, ["b"]);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.apply(undefined, ["b", 4]);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.apply(null);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.apply(null, ["b"]);
}, TypeError);
assertThrows(function() {
  String.prototype.startsWith.apply(null, ["b", 4]);
}, TypeError);
assertEquals(String.prototype.startsWith.apply(42, ["2"]), false);
assertEquals(String.prototype.startsWith.apply(42, ["4"]), true);
assertEquals(String.prototype.startsWith.apply(42, ["b", 4]), false);
assertEquals(String.prototype.startsWith.apply(42, ["2", 1]), true);
assertEquals(String.prototype.startsWith.apply(42, ["2", 4]), false);
assertEquals(String.prototype.startsWith.apply({
  "toString": function() {
    return "abc";
  }
}, ["b", 0]), false);
assertEquals(String.prototype.startsWith.apply({
  "toString": function() {
    return "abc";
  }
}, ["b", 1]), true);
assertEquals(String.prototype.startsWith.apply({
  "toString": function() {
    return "abc";
  }
}, ["b", 2]), false);
assertThrows(function() {
  String.prototype.startsWith.apply({
    "toString": function() { throw RangeError(); }
  }, [/./]);
}, RangeError);
assertThrows(function() {
  String.prototype.startsWith.apply({
    "toString": function() { return "abc"; }
  }, [/./]);
}, TypeError);

// startsWith does its brand checks with Symbol.match
var re = /./;
assertThrows(function() {
  "".startsWith(re);
}, TypeError);
re[Symbol.match] = false;
assertEquals(false, "".startsWith(re));