michael@0: /* michael@0: * Copyright (c) 2013 Mathias Bynens. All rights reserved. michael@0: * michael@0: * Permission is hereby granted, free of charge, to any person obtaining a michael@0: * copy of this software and associated documentation files (the "Software"), michael@0: * to deal in the Software without restriction, including without limitation michael@0: * the rights to use, copy, modify, merge, publish, distribute, sublicense, michael@0: * and/or sell copies of the Software, and to permit persons to whom the michael@0: * Software is furnished to do so, subject to the following conditions: michael@0: * michael@0: * The above copyright notice and this permission notice shall be included in michael@0: * all copies or substantial portions of the Software. michael@0: * michael@0: * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR michael@0: * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, michael@0: * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE michael@0: * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER michael@0: * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING michael@0: * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER michael@0: * DEALINGS IN THE SOFTWARE. michael@0: */ michael@0: michael@0: function assertThrows(fun, errorType) { michael@0: try { michael@0: fun(); michael@0: assertEq(true, false, "Expected error, but none was thrown"); michael@0: } catch (e) { michael@0: assertEq(e instanceof errorType, true, "Wrong error type thrown"); michael@0: } michael@0: } michael@0: michael@0: Object.prototype[1] = 2; // try to break `arguments[1]` michael@0: michael@0: assertEq(String.prototype.startsWith.length, 1); michael@0: assertEq(String.prototype.propertyIsEnumerable('startsWith'), false); michael@0: michael@0: assertEq('undefined'.startsWith(), true); michael@0: assertEq('undefined'.startsWith(undefined), true); michael@0: assertEq('undefined'.startsWith(null), false); michael@0: assertEq('null'.startsWith(), false); michael@0: assertEq('null'.startsWith(undefined), false); michael@0: assertEq('null'.startsWith(null), true); michael@0: michael@0: assertEq('abc'.startsWith(), false); michael@0: assertEq('abc'.startsWith(''), true); michael@0: assertEq('abc'.startsWith('\0'), false); michael@0: assertEq('abc'.startsWith('a'), true); michael@0: assertEq('abc'.startsWith('b'), false); michael@0: assertEq('abc'.startsWith('ab'), true); michael@0: assertEq('abc'.startsWith('bc'), false); michael@0: assertEq('abc'.startsWith('abc'), true); michael@0: assertEq('abc'.startsWith('bcd'), false); michael@0: assertEq('abc'.startsWith('abcd'), false); michael@0: assertEq('abc'.startsWith('bcde'), false); michael@0: michael@0: assertEq('abc'.startsWith('', NaN), true); michael@0: assertEq('abc'.startsWith('\0', NaN), false); michael@0: assertEq('abc'.startsWith('a', NaN), true); michael@0: assertEq('abc'.startsWith('b', NaN), false); michael@0: assertEq('abc'.startsWith('ab', NaN), true); michael@0: assertEq('abc'.startsWith('bc', NaN), false); michael@0: assertEq('abc'.startsWith('abc', NaN), true); michael@0: assertEq('abc'.startsWith('bcd', NaN), false); michael@0: assertEq('abc'.startsWith('abcd', NaN), false); michael@0: assertEq('abc'.startsWith('bcde', NaN), false); michael@0: michael@0: assertEq('abc'.startsWith('', false), true); michael@0: assertEq('abc'.startsWith('\0', false), false); michael@0: assertEq('abc'.startsWith('a', false), true); michael@0: assertEq('abc'.startsWith('b', false), false); michael@0: assertEq('abc'.startsWith('ab', false), true); michael@0: assertEq('abc'.startsWith('bc', false), false); michael@0: assertEq('abc'.startsWith('abc', false), true); michael@0: assertEq('abc'.startsWith('bcd', false), false); michael@0: assertEq('abc'.startsWith('abcd', false), false); michael@0: assertEq('abc'.startsWith('bcde', false), false); michael@0: michael@0: assertEq('abc'.startsWith('', undefined), true); michael@0: assertEq('abc'.startsWith('\0', undefined), false); michael@0: assertEq('abc'.startsWith('a', undefined), true); michael@0: assertEq('abc'.startsWith('b', undefined), false); michael@0: assertEq('abc'.startsWith('ab', undefined), true); michael@0: assertEq('abc'.startsWith('bc', undefined), false); michael@0: assertEq('abc'.startsWith('abc', undefined), true); michael@0: assertEq('abc'.startsWith('bcd', undefined), false); michael@0: assertEq('abc'.startsWith('abcd', undefined), false); michael@0: assertEq('abc'.startsWith('bcde', undefined), false); michael@0: michael@0: assertEq('abc'.startsWith('', null), true); michael@0: assertEq('abc'.startsWith('\0', null), false); michael@0: assertEq('abc'.startsWith('a', null), true); michael@0: assertEq('abc'.startsWith('b', null), false); michael@0: assertEq('abc'.startsWith('ab', null), true); michael@0: assertEq('abc'.startsWith('bc', null), false); michael@0: assertEq('abc'.startsWith('abc', null), true); michael@0: assertEq('abc'.startsWith('bcd', null), false); michael@0: assertEq('abc'.startsWith('abcd', null), false); michael@0: assertEq('abc'.startsWith('bcde', null), false); michael@0: michael@0: assertEq('abc'.startsWith('', -Infinity), true); michael@0: assertEq('abc'.startsWith('\0', -Infinity), false); michael@0: assertEq('abc'.startsWith('a', -Infinity), true); michael@0: assertEq('abc'.startsWith('b', -Infinity), false); michael@0: assertEq('abc'.startsWith('ab', -Infinity), true); michael@0: assertEq('abc'.startsWith('bc', -Infinity), false); michael@0: assertEq('abc'.startsWith('abc', -Infinity), true); michael@0: assertEq('abc'.startsWith('bcd', -Infinity), false); michael@0: assertEq('abc'.startsWith('abcd', -Infinity), false); michael@0: assertEq('abc'.startsWith('bcde', -Infinity), false); michael@0: michael@0: assertEq('abc'.startsWith('', -1), true); michael@0: assertEq('abc'.startsWith('\0', -1), false); michael@0: assertEq('abc'.startsWith('a', -1), true); michael@0: assertEq('abc'.startsWith('b', -1), false); michael@0: assertEq('abc'.startsWith('ab', -1), true); michael@0: assertEq('abc'.startsWith('bc', -1), false); michael@0: assertEq('abc'.startsWith('abc', -1), true); michael@0: assertEq('abc'.startsWith('bcd', -1), false); michael@0: assertEq('abc'.startsWith('abcd', -1), false); michael@0: assertEq('abc'.startsWith('bcde', -1), false); michael@0: michael@0: assertEq('abc'.startsWith('', -0), true); michael@0: assertEq('abc'.startsWith('\0', -0), false); michael@0: assertEq('abc'.startsWith('a', -0), true); michael@0: assertEq('abc'.startsWith('b', -0), false); michael@0: assertEq('abc'.startsWith('ab', -0), true); michael@0: assertEq('abc'.startsWith('bc', -0), false); michael@0: assertEq('abc'.startsWith('abc', -0), true); michael@0: assertEq('abc'.startsWith('bcd', -0), false); michael@0: assertEq('abc'.startsWith('abcd', -0), false); michael@0: assertEq('abc'.startsWith('bcde', -0), false); michael@0: michael@0: assertEq('abc'.startsWith('', +0), true); michael@0: assertEq('abc'.startsWith('\0', +0), false); michael@0: assertEq('abc'.startsWith('a', +0), true); michael@0: assertEq('abc'.startsWith('b', +0), false); michael@0: assertEq('abc'.startsWith('ab', +0), true); michael@0: assertEq('abc'.startsWith('bc', +0), false); michael@0: assertEq('abc'.startsWith('abc', +0), true); michael@0: assertEq('abc'.startsWith('bcd', +0), false); michael@0: assertEq('abc'.startsWith('abcd', +0), false); michael@0: assertEq('abc'.startsWith('bcde', +0), false); michael@0: michael@0: assertEq('abc'.startsWith('', 1), true); michael@0: assertEq('abc'.startsWith('\0', 1), false); michael@0: assertEq('abc'.startsWith('a', 1), false); michael@0: assertEq('abc'.startsWith('b', 1), true); michael@0: assertEq('abc'.startsWith('ab', 1), false); michael@0: assertEq('abc'.startsWith('bc', 1), true); michael@0: assertEq('abc'.startsWith('abc', 1), false); michael@0: assertEq('abc'.startsWith('bcd', 1), false); michael@0: assertEq('abc'.startsWith('abcd', 1), false); michael@0: assertEq('abc'.startsWith('bcde', 1), false); michael@0: michael@0: assertEq('abc'.startsWith('', +Infinity), true); michael@0: assertEq('abc'.startsWith('\0', +Infinity), false); michael@0: assertEq('abc'.startsWith('a', +Infinity), false); michael@0: assertEq('abc'.startsWith('b', +Infinity), false); michael@0: assertEq('abc'.startsWith('ab', +Infinity), false); michael@0: assertEq('abc'.startsWith('bc', +Infinity), false); michael@0: assertEq('abc'.startsWith('abc', +Infinity), false); michael@0: assertEq('abc'.startsWith('bcd', +Infinity), false); michael@0: assertEq('abc'.startsWith('abcd', +Infinity), false); michael@0: assertEq('abc'.startsWith('bcde', +Infinity), false); michael@0: michael@0: assertEq('abc'.startsWith('', true), true); michael@0: assertEq('abc'.startsWith('\0', true), false); michael@0: assertEq('abc'.startsWith('a', true), false); michael@0: assertEq('abc'.startsWith('b', true), true); michael@0: assertEq('abc'.startsWith('ab', true), false); michael@0: assertEq('abc'.startsWith('bc', true), true); michael@0: assertEq('abc'.startsWith('abc', true), false); michael@0: assertEq('abc'.startsWith('bcd', true), false); michael@0: assertEq('abc'.startsWith('abcd', true), false); michael@0: assertEq('abc'.startsWith('bcde', true), false); michael@0: michael@0: assertEq('abc'.startsWith('', 'x'), true); michael@0: assertEq('abc'.startsWith('\0', 'x'), false); michael@0: assertEq('abc'.startsWith('a', 'x'), true); michael@0: assertEq('abc'.startsWith('b', 'x'), false); michael@0: assertEq('abc'.startsWith('ab', 'x'), true); michael@0: assertEq('abc'.startsWith('bc', 'x'), false); michael@0: assertEq('abc'.startsWith('abc', 'x'), true); michael@0: assertEq('abc'.startsWith('bcd', 'x'), false); michael@0: assertEq('abc'.startsWith('abcd', 'x'), false); michael@0: assertEq('abc'.startsWith('bcde', 'x'), false); michael@0: michael@0: assertEq('[a-z]+(bar)?'.startsWith('[a-z]+'), true); michael@0: assertThrows(function() { '[a-z]+(bar)?'.startsWith(/[a-z]+/); }, TypeError); michael@0: assertEq('[a-z]+(bar)?'.startsWith('(bar)?', 6), true); michael@0: assertThrows(function() { '[a-z]+(bar)?'.startsWith(/(bar)?/); }, TypeError); michael@0: assertThrows(function() { '[a-z]+/(bar)?/'.startsWith(/(bar)?/); }, TypeError); michael@0: var global = newGlobal(); michael@0: global.eval('this.re = /(bar)?/'); michael@0: assertThrows(function() { '[a-z]+/(bar)?/'.startsWith(global.re); }, TypeError); michael@0: michael@0: // http://mathiasbynens.be/notes/javascript-unicode#poo-test michael@0: var string = 'I\xF1t\xEBrn\xE2ti\xF4n\xE0liz\xE6ti\xF8n\u2603\uD83D\uDCA9'; michael@0: assertEq(string.startsWith(''), true); michael@0: assertEq(string.startsWith('\xF1t\xEBr'), false); michael@0: assertEq(string.startsWith('\xF1t\xEBr', 1), true); michael@0: assertEq(string.startsWith('\xE0liz\xE6'), false); michael@0: assertEq(string.startsWith('\xE0liz\xE6', 11), true); michael@0: assertEq(string.startsWith('\xF8n\u2603\uD83D\uDCA9'), false); michael@0: assertEq(string.startsWith('\xF8n\u2603\uD83D\uDCA9', 18), true); michael@0: assertEq(string.startsWith('\u2603'), false); michael@0: assertEq(string.startsWith('\u2603', 20), true); michael@0: assertEq(string.startsWith('\uD83D\uDCA9'), false); michael@0: assertEq(string.startsWith('\uD83D\uDCA9', 21), true); michael@0: michael@0: assertThrows(function() { String.prototype.startsWith.call(undefined); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.call(undefined, 'b'); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.call(undefined, 'b', 4); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.call(null); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.call(null, 'b'); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.call(null, 'b', 4); }, TypeError); michael@0: assertEq(String.prototype.startsWith.call(42, '2'), false); michael@0: assertEq(String.prototype.startsWith.call(42, '4'), true); michael@0: assertEq(String.prototype.startsWith.call(42, 'b', 4), false); michael@0: assertEq(String.prototype.startsWith.call(42, '2', 1), true); michael@0: assertEq(String.prototype.startsWith.call(42, '2', 4), false); michael@0: assertEq(String.prototype.startsWith.call({ 'toString': function() { return 'abc'; } }, 'b', 0), false); michael@0: assertEq(String.prototype.startsWith.call({ 'toString': function() { return 'abc'; } }, 'b', 1), true); michael@0: assertEq(String.prototype.startsWith.call({ 'toString': function() { return 'abc'; } }, 'b', 2), false); michael@0: assertThrows(function() { String.prototype.startsWith.call({ 'toString': function() { throw RangeError(); } }, /./); }, RangeError); michael@0: assertThrows(function() { String.prototype.startsWith.call({ 'toString': function() { return 'abc'; } }, /./); }, TypeError); michael@0: michael@0: assertThrows(function() { String.prototype.startsWith.apply(undefined); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.apply(undefined, ['b']); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.apply(undefined, ['b', 4]); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.apply(null); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.apply(null, ['b']); }, TypeError); michael@0: assertThrows(function() { String.prototype.startsWith.apply(null, ['b', 4]); }, TypeError); michael@0: assertEq(String.prototype.startsWith.apply(42, ['2']), false); michael@0: assertEq(String.prototype.startsWith.apply(42, ['4']), true); michael@0: assertEq(String.prototype.startsWith.apply(42, ['b', 4]), false); michael@0: assertEq(String.prototype.startsWith.apply(42, ['2', 1]), true); michael@0: assertEq(String.prototype.startsWith.apply(42, ['2', 4]), false); michael@0: assertEq(String.prototype.startsWith.apply({ 'toString': function() { return 'abc'; } }, ['b', 0]), false); michael@0: assertEq(String.prototype.startsWith.apply({ 'toString': function() { return 'abc'; } }, ['b', 1]), true); michael@0: assertEq(String.prototype.startsWith.apply({ 'toString': function() { return 'abc'; } }, ['b', 2]), false); michael@0: assertThrows(function() { String.prototype.startsWith.apply({ 'toString': function() { throw RangeError(); } }, [/./]); }, RangeError); michael@0: assertThrows(function() { String.prototype.startsWith.apply({ 'toString': function() { return 'abc'; } }, [/./]); }, TypeError);