|  | // Copyright 2013 the V8 project authors. All rights reserved. | 
|  | // Copyright (C) 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved. | 
|  | // | 
|  | // Redistribution and use in source and binary forms, with or without | 
|  | // modification, are permitted provided that the following conditions | 
|  | // are met: | 
|  | // 1.  Redistributions of source code must retain the above copyright | 
|  | //     notice, this list of conditions and the following disclaimer. | 
|  | // 2.  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. | 
|  | // | 
|  | // THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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. | 
|  |  | 
|  | description( | 
|  | "This test checks whether various seal/freeze/preventExtentions work on a regular object." | 
|  | ); | 
|  |  | 
|  | function obj() | 
|  | { | 
|  | // Add an accessor property to check 'isFrozen' returns the correct result for objects with accessors. | 
|  | return Object.defineProperty({ a: 1, b: 2 }, 'g', { get: function() { return "getter"; } }); | 
|  | } | 
|  |  | 
|  | function test(obj) | 
|  | { | 
|  | obj.c =3; | 
|  | obj.b =4; | 
|  | delete obj.a; | 
|  |  | 
|  | var result = ""; | 
|  | for (key in obj) | 
|  | result += ("(" + key + ":" + obj[key] + ")"); | 
|  | if (Object.isSealed(obj)) | 
|  | result += "S"; | 
|  | if (Object.isFrozen(obj)) | 
|  | result += "F"; | 
|  | if (Object.isExtensible(obj)) | 
|  | result += "E"; | 
|  | return result; | 
|  | } | 
|  |  | 
|  | function seal(obj) | 
|  | { | 
|  | Object.seal(obj); | 
|  | return obj; | 
|  | } | 
|  |  | 
|  | function freeze(obj) | 
|  | { | 
|  | Object.freeze(obj); | 
|  | return obj; | 
|  | } | 
|  |  | 
|  | function preventExtensions(obj) | 
|  | { | 
|  | Object.preventExtensions(obj); | 
|  | return obj; | 
|  | } | 
|  |  | 
|  | function inextensible(){} | 
|  | function sealed(){} | 
|  | function frozen(){}; | 
|  | preventExtensions(inextensible); | 
|  | seal(sealed); | 
|  | freeze(frozen); | 
|  | new inextensible; | 
|  | new sealed; | 
|  | new frozen; | 
|  | inextensible.prototype.prototypeExists = true; | 
|  | sealed.prototype.prototypeExists = true; | 
|  | frozen.prototype.prototypeExists = true; | 
|  |  | 
|  | shouldBeTrue("(new inextensible).prototypeExists"); | 
|  | shouldBeTrue("(new sealed).prototypeExists"); | 
|  | shouldBeTrue("(new frozen).prototypeExists"); | 
|  |  | 
|  | shouldBe('test(obj())', '"(b:4)(c:3)E"'); // extensible, can delete a, can modify b, and can add c | 
|  | shouldBe('test(preventExtensions(obj()))', '"(b:4)"'); // <nothing>, can delete a, can modify b, and CANNOT add c | 
|  | shouldBe('test(seal(obj()))', '"(a:1)(b:4)S"'); // sealed, CANNOT delete a, can modify b, and CANNOT add c | 
|  | shouldBe('test(freeze(obj()))', '"(a:1)(b:2)SF"'); // sealed and frozen, CANNOT delete a, CANNOT modify b, and CANNOT add c | 
|  |  | 
|  | // check that we can preventExtensions on a host function. | 
|  | shouldBe('Object.preventExtensions(Math.sin)', 'Math.sin'); | 
|  |  | 
|  | shouldThrow('var o = {}; Object.preventExtensions(o); o.__proto__ = { newProp: "Should not see this" }; o.newProp;'); | 
|  | shouldThrow('"use strict"; var o = {}; Object.preventExtensions(o); o.__proto__ = { newProp: "Should not see this" }; o.newProp;'); | 
|  |  | 
|  | // check that we can still access static properties on an object after calling preventExtensions. | 
|  | shouldBe('Object.preventExtensions(Math); Math.sqrt(4)', '2'); | 
|  |  | 
|  | // Should not be able to add properties to a preventExtensions array. | 
|  | shouldBeUndefined('var arr = Object.preventExtensions([]); arr[0] = 42; arr[0]'); | 
|  | shouldBe('var arr = Object.preventExtensions([]); arr[0] = 42; arr.length', '0'); | 
|  | // In strict mode, this throws. | 
|  | shouldThrow('"use strict"; var arr = Object.preventExtensions([]); arr[0] = 42; arr[0]'); | 
|  |  | 
|  | // A read-only property on the prototype should prevent a [[Put]] . | 
|  | function Constructor() {} | 
|  | Constructor.prototype.foo = 1; | 
|  | Object.freeze(Constructor.prototype); | 
|  | var obj = new Constructor(); | 
|  | obj.foo = 2; | 
|  | shouldBe('obj.foo', '1'); | 
|  |  | 
|  | // Check that freezing a function works correctly. | 
|  | var func = freeze(function foo(){}); | 
|  | shouldBeTrue('Object.isFrozen(func)') | 
|  | func.prototype = 42; | 
|  | shouldBeFalse('func.prototype === 42'); | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(func, "prototype").writable') | 
|  |  | 
|  | // Check that freezing a strict function works correctly. | 
|  | var strictFunc = freeze(function foo(){ "use strict"; }); | 
|  | shouldBeTrue('Object.isFrozen(strictFunc)') | 
|  | strictFunc.prototype = 42; | 
|  | shouldBeFalse('strictFunc.prototype === 42'); | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(strictFunc, "prototype").writable') | 
|  |  | 
|  | // Check that freezing array objects works correctly. | 
|  | var array = freeze([0,1,2]); | 
|  | shouldBeTrue('Object.isFrozen(array)') | 
|  | array[0] = 3; | 
|  | shouldBe('array[0]', '0'); | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(array, "length").writable') | 
|  |  | 
|  | // Check that freezing arguments objects works correctly. | 
|  | var args = freeze((function(){ return arguments; })(0,1,2)); | 
|  | shouldBeTrue('Object.isFrozen(args)') | 
|  | args[0] = 3; | 
|  | shouldBe('args[0]', '0'); | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(args, "length").writable') | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(args, "callee").writable') | 
|  |  | 
|  | // Check that freeze still works if preventExtensions has been called on the object. | 
|  | function preventExtensionsFreezeIsFrozen(x) | 
|  | { | 
|  | Object.preventExtensions(x); | 
|  | Object.freeze(x); | 
|  | return Object.isFrozen(x); | 
|  | } | 
|  | shouldBeTrue('preventExtensionsFreezeIsFrozen(function foo(){})') | 
|  | shouldBeTrue('preventExtensionsFreezeIsFrozen(function foo(){ "use strict"; })') | 
|  | shouldBeTrue('preventExtensionsFreezeIsFrozen([0,1,2])') | 
|  | shouldBeTrue('preventExtensionsFreezeIsFrozen((function(){ return arguments; })(0,1,2))') | 
|  |  | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(freeze({0:0}), 0).configurable'); | 
|  | shouldBeFalse('Object.getOwnPropertyDescriptor(freeze({10000001:0}), 10000001).configurable'); |