// 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("Test behaviour of JSON reviver function.") if (!Array.isArray) Array.isArray = function(o) { return o.constructor === Array; } function arrayReviver(i,v) { if (i != "") { currentHolder = this; debug(""); debug("Ensure the holder for our array is indeed an array"); shouldBeTrue("Array.isArray(currentHolder)"); shouldBe("currentHolder.length", "" + expectedLength); if (i > 0) { debug(""); debug("Ensure that we always get the same holder"); shouldBe("currentHolder", "lastHolder"); } switch (Number(i)) { case 0: v = undefined; debug(""); debug("Ensure that the holder already has all the properties present at the start of filtering"); shouldBe("currentHolder[0]", '"a value"'); shouldBe("currentHolder[1]", '"another value"'); shouldBe("currentHolder[2]", '"and another value"'); shouldBe("currentHolder[3]", '"to delete"'); shouldBe("currentHolder[4]", '"extra value"'); break; case 1: debug(""); debug("Ensure that returning undefined has removed the property 0 from the holder during filtering."); shouldBeFalse("currentHolder.hasOwnProperty(0)"); currentHolder[2] = "a replaced value"; break; case 2: debug(""); debug("Ensure that changing the value of a property is reflected while filtering.") shouldBe("currentHolder[2]", '"a replaced value"'); value = v; debug(""); debug("Ensure that the changed value is reflected in the arguments passed to the reviver"); shouldBe("value", "currentHolder[2]"); delete this[3]; break; case 3: debug(""); debug("Ensure that we visited a value that we have deleted, and that deletion is reflected while filtering."); shouldBeFalse("currentHolder.hasOwnProperty(3)"); value = v; debug(""); debug("Ensure that when visiting a deleted property value is undefined"); shouldBeUndefined("value"); this.length = 3; v = "undelete the property"; expectedLength = 4; break; case 4: if (this.length != 4) { testFailed("Did not call reviver for deleted property"); expectedLength = this.length = 3; break; } case 5: testPassed("Ensured that property was visited despite Array length being reduced."); value = v; shouldBeUndefined("value"); this[10] = "fail"; break; default: testFailed("Visited unexpected property " + i + " with value " + v); } } lastHolder = this; return v; } expectedLength = 5; var result = JSON.parse('["a value", "another value", "and another value", "to delete", "extra value"]', arrayReviver); debug(""); debug("Ensure that we created the root holder as specified in ES5"); shouldBeTrue("'' in lastHolder"); shouldBe("result", "lastHolder['']"); debug(""); debug("Ensure that a deleted value is revived if the reviver function returns a value"); shouldBeTrue("result.hasOwnProperty(3)"); function objectReviver(i,v) { if (i != "") { currentHolder = this; shouldBeTrue("currentHolder != globalObject"); if (seen) { debug(""); debug("Ensure that we get the same holder object for each property"); shouldBe("currentHolder", "lastHolder"); } seen = true; switch (i) { case "a property": v = undefined; debug(""); debug("Ensure that the holder already has all the properties present at the start of filtering"); shouldBe("currentHolder['a property']", '"a value"'); shouldBe("currentHolder['another property']", '"another value"'); shouldBe("currentHolder['and another property']", '"and another value"'); shouldBe("currentHolder['to delete']", '"will be deleted"'); break; case "another property": debug(""); debug("Ensure that returning undefined has correctly removed the property 'a property' from the holder object"); shouldBeFalse("currentHolder.hasOwnProperty('a property')"); currentHolder['and another property'] = "a replaced value"; break; case "and another property": debug("Ensure that changing the value of a property is reflected while filtering."); shouldBe("currentHolder['and another property']", '"a replaced value"'); value = v; debug(""); debug("Ensure that the changed value is reflected in the arguments passed to the reviver"); shouldBe("value", '"a replaced value"'); delete this["to delete"]; break; case "to delete": debug(""); debug("Ensure that we visited a value that we have deleted, and that deletion is reflected while filtering."); shouldBeFalse("currentHolder.hasOwnProperty('to delete')"); value = v; debug(""); debug("Ensure that when visiting a deleted property value is undefined"); shouldBeUndefined("value"); v = "undelete the property"; this["new property"] = "fail"; break; default: testFailed("Visited unexpected property " + i + " with value " + v); } } lastHolder = this; return v; } debug(""); debug("Test behaviour of revivor used in conjunction with an object"); var seen = false; var globalObject = this; var result = JSON.parse('{"a property" : "a value", "another property" : "another value", "and another property" : "and another value", "to delete" : "will be deleted"}', objectReviver); debug(""); debug("Ensure that we created the root holder as specified in ES5"); shouldBeTrue("lastHolder.hasOwnProperty('')"); shouldBeFalse("result.hasOwnProperty('a property')"); shouldBeTrue("result.hasOwnProperty('to delete')"); shouldBe("result", "lastHolder['']"); debug(""); debug("Test behaviour of revivor that introduces a cycle"); function reviveAddsCycle(i, v) { if (i == 0) this[1] = this; return v; } shouldThrow('JSON.parse("[0,1]", reviveAddsCycle)'); debug(""); debug("Test behaviour of revivor that introduces a new array classed object (the result of a regex)"); var createdBadness = false; function reviveIntroducesNewArrayLikeObject(i, v) { if (i == 0 && !createdBadness) { this[1] = /(a)+/.exec("a"); createdBadness = true; } return v; } shouldBe('JSON.stringify(JSON.parse("[0,1]", reviveIntroducesNewArrayLikeObject))', '\'[0,["a","a"]]\'');