|
1 /* |
|
2 * Copyright (C) 2006 Apple Computer, Inc. All rights reserved. |
|
3 * |
|
4 * Redistribution and use in source and binary forms, with or without |
|
5 * modification, are permitted provided that the following conditions |
|
6 * are met: |
|
7 * 1. Redistributions of source code must retain the above copyright |
|
8 * notice, this list of conditions and the following disclaimer. |
|
9 * 2. Redistributions in binary form must reproduce the above copyright |
|
10 * notice, this list of conditions and the following disclaimer in the |
|
11 * documentation and/or other materials provided with the distribution. |
|
12 * |
|
13 * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY |
|
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE |
|
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
|
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR |
|
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, |
|
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
|
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR |
|
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY |
|
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT |
|
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE |
|
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
|
24 */ |
|
25 |
|
26 function bludgeonArguments() { if (0) arguments; return function g() {} } |
|
27 h = bludgeonArguments(); |
|
28 gc(); |
|
29 |
|
30 var failed = false; |
|
31 function pass(msg) |
|
32 { |
|
33 print("PASS: " + msg, "green"); |
|
34 } |
|
35 |
|
36 function fail(msg) |
|
37 { |
|
38 print("FAIL: " + msg, "red"); |
|
39 failed = true; |
|
40 } |
|
41 |
|
42 function shouldBe(a, b) |
|
43 { |
|
44 var evalA; |
|
45 try { |
|
46 evalA = eval(a); |
|
47 } catch(e) { |
|
48 evalA = e; |
|
49 } |
|
50 |
|
51 if (evalA == b || isNaN(evalA) && typeof evalA == 'number' && isNaN(b) && typeof b == 'number') |
|
52 pass(a + " should be " + b + " and is."); |
|
53 else |
|
54 fail(a + " should be " + b + " but instead is " + evalA + "."); |
|
55 } |
|
56 |
|
57 function shouldThrow(a) |
|
58 { |
|
59 var evalA; |
|
60 try { |
|
61 eval(a); |
|
62 } catch(e) { |
|
63 pass(a + " threw: " + e); |
|
64 return; |
|
65 } |
|
66 |
|
67 fail(a + " did not throw an exception."); |
|
68 } |
|
69 |
|
70 function globalStaticFunction() |
|
71 { |
|
72 return 4; |
|
73 } |
|
74 |
|
75 shouldBe("globalStaticValue", 3); |
|
76 shouldBe("globalStaticFunction()", 4); |
|
77 |
|
78 shouldBe("typeof MyObject", "function"); // our object implements 'call' |
|
79 MyObject.cantFind = 1; |
|
80 shouldBe("MyObject.cantFind", undefined); |
|
81 MyObject.regularType = 1; |
|
82 shouldBe("MyObject.regularType", 1); |
|
83 MyObject.alwaysOne = 2; |
|
84 shouldBe("MyObject.alwaysOne", 1); |
|
85 MyObject.cantDelete = 1; |
|
86 delete MyObject.cantDelete; |
|
87 shouldBe("MyObject.cantDelete", 1); |
|
88 shouldBe("delete MyObject.throwOnDelete", "an exception"); |
|
89 MyObject.cantSet = 1; |
|
90 shouldBe("MyObject.cantSet", undefined); |
|
91 shouldBe("MyObject.throwOnGet", "an exception"); |
|
92 shouldBe("MyObject.throwOnSet = 5", "an exception"); |
|
93 shouldBe("MyObject('throwOnCall')", "an exception"); |
|
94 shouldBe("new MyObject('throwOnConstruct')", "an exception"); |
|
95 shouldBe("'throwOnHasInstance' instanceof MyObject", "an exception"); |
|
96 |
|
97 var foundMyPropertyName = false; |
|
98 var foundRegularType = false; |
|
99 for (var p in MyObject) { |
|
100 if (p == "myPropertyName") |
|
101 foundMyPropertyName = true; |
|
102 if (p == "regularType") |
|
103 foundRegularType = true; |
|
104 } |
|
105 |
|
106 if (foundMyPropertyName) |
|
107 pass("MyObject.myPropertyName was enumerated"); |
|
108 else |
|
109 fail("MyObject.myPropertyName was not enumerated"); |
|
110 |
|
111 if (foundRegularType) |
|
112 pass("MyObject.regularType was enumerated"); |
|
113 else |
|
114 fail("MyObject.regularType was not enumerated"); |
|
115 |
|
116 var alwaysOneDescriptor = Object.getOwnPropertyDescriptor(MyObject, "alwaysOne"); |
|
117 shouldBe('typeof alwaysOneDescriptor', "object"); |
|
118 shouldBe('alwaysOneDescriptor.value', MyObject.alwaysOne); |
|
119 shouldBe('alwaysOneDescriptor.configurable', true); |
|
120 shouldBe('alwaysOneDescriptor.enumerable', false); // Actually it is. |
|
121 var cantFindDescriptor = Object.getOwnPropertyDescriptor(MyObject, "cantFind"); |
|
122 shouldBe('typeof cantFindDescriptor', "object"); |
|
123 shouldBe('cantFindDescriptor.value', MyObject.cantFind); |
|
124 shouldBe('cantFindDescriptor.configurable', true); |
|
125 shouldBe('cantFindDescriptor.enumerable', false); |
|
126 try { |
|
127 // If getOwnPropertyDescriptor() returned an access descriptor, this wouldn't throw. |
|
128 Object.getOwnPropertyDescriptor(MyObject, "throwOnGet"); |
|
129 } catch (e) { |
|
130 pass("getting property descriptor of throwOnGet threw exception"); |
|
131 } |
|
132 var myPropertyNameDescriptor = Object.getOwnPropertyDescriptor(MyObject, "myPropertyName"); |
|
133 shouldBe('typeof myPropertyNameDescriptor', "object"); |
|
134 shouldBe('myPropertyNameDescriptor.value', MyObject.myPropertyName); |
|
135 shouldBe('myPropertyNameDescriptor.configurable', true); |
|
136 shouldBe('myPropertyNameDescriptor.enumerable', false); // Actually it is. |
|
137 try { |
|
138 // if getOwnPropertyDescriptor() returned an access descriptor, this wouldn't throw. |
|
139 Object.getOwnPropertyDescriptor(MyObject, "hasPropertyLie"); |
|
140 } catch (e) { |
|
141 pass("getting property descriptor of hasPropertyLie threw exception"); |
|
142 } |
|
143 shouldBe('Object.getOwnPropertyDescriptor(MyObject, "doesNotExist")', undefined); |
|
144 |
|
145 myObject = new MyObject(); |
|
146 |
|
147 shouldBe("delete MyObject.regularType", true); |
|
148 shouldBe("MyObject.regularType", undefined); |
|
149 shouldBe("MyObject(0)", 1); |
|
150 shouldBe("MyObject()", undefined); |
|
151 shouldBe("typeof myObject", "object"); |
|
152 shouldBe("MyObject ? 1 : 0", true); // toBoolean |
|
153 shouldBe("+MyObject", 1); // toNumber |
|
154 shouldBe("(MyObject.toString())", "[object MyObject]"); // toString |
|
155 shouldBe("String(MyObject)", "MyObjectAsString"); // type conversion to string |
|
156 shouldBe("MyObject - 0", 1); // toNumber |
|
157 |
|
158 shouldBe("typeof MyConstructor", "object"); |
|
159 constructedObject = new MyConstructor(1); |
|
160 shouldBe("typeof constructedObject", "object"); |
|
161 shouldBe("constructedObject.value", 1); |
|
162 shouldBe("myObject instanceof MyObject", true); |
|
163 shouldBe("(new Object()) instanceof MyObject", false); |
|
164 |
|
165 shouldThrow("MyObject.nullGetSet = 1"); |
|
166 shouldThrow("MyObject.nullGetSet"); |
|
167 shouldThrow("MyObject.nullCall()"); |
|
168 shouldThrow("MyObject.hasPropertyLie"); |
|
169 |
|
170 derived = new Derived(); |
|
171 |
|
172 shouldBe("derived instanceof Derived", true); |
|
173 shouldBe("derived instanceof Base", true); |
|
174 |
|
175 // base properties and functions return 1 when called/gotten; derived, 2 |
|
176 shouldBe("derived.baseProtoDup()", 2); |
|
177 shouldBe("derived.baseProto()", 1); |
|
178 shouldBe("derived.baseDup", 2); |
|
179 shouldBe("derived.baseOnly", 1); |
|
180 shouldBe("derived.protoOnly()", 2); |
|
181 shouldBe("derived.protoDup", 2); |
|
182 shouldBe("derived.derivedOnly", 2) |
|
183 |
|
184 // base properties throw 1 when set; derived, 2 |
|
185 shouldBe("derived.baseDup = 0", 2); |
|
186 shouldBe("derived.baseOnly = 0", 1); |
|
187 shouldBe("derived.derivedOnly = 0", 2) |
|
188 shouldBe("derived.protoDup = 0", 2); |
|
189 |
|
190 derived2 = new Derived2(); |
|
191 |
|
192 shouldBe("derived2 instanceof Derived2", true); |
|
193 shouldBe("derived2 instanceof Derived", true); |
|
194 shouldBe("derived2 instanceof Base", true); |
|
195 |
|
196 // base properties and functions return 1 when called/gotten; derived, 2 |
|
197 shouldBe("derived2.baseProtoDup()", 2); |
|
198 shouldBe("derived2.baseProto()", 1); |
|
199 shouldBe("derived2.baseDup", 2); |
|
200 shouldBe("derived2.baseOnly", 1); |
|
201 shouldBe("derived2.protoOnly()", 2); |
|
202 shouldBe("derived2.protoDup", 2); |
|
203 shouldBe("derived2.derivedOnly", 2) |
|
204 |
|
205 // base properties throw 1 when set; derived, 2 |
|
206 shouldBe("derived2.baseDup = 0", 2); |
|
207 shouldBe("derived2.baseOnly = 0", 1); |
|
208 shouldBe("derived2.derivedOnly = 0", 2) |
|
209 shouldBe("derived2.protoDup = 0", 2); |
|
210 |
|
211 shouldBe('Object.getOwnPropertyDescriptor(derived, "baseProto")', undefined); |
|
212 shouldBe('Object.getOwnPropertyDescriptor(derived, "baseProtoDup")', undefined); |
|
213 var baseDupDescriptor = Object.getOwnPropertyDescriptor(derived, "baseDup"); |
|
214 shouldBe('typeof baseDupDescriptor', "object"); |
|
215 shouldBe('baseDupDescriptor.value', derived.baseDup); |
|
216 shouldBe('baseDupDescriptor.configurable', true); |
|
217 shouldBe('baseDupDescriptor.enumerable', false); |
|
218 var baseOnlyDescriptor = Object.getOwnPropertyDescriptor(derived, "baseOnly"); |
|
219 shouldBe('typeof baseOnlyDescriptor', "object"); |
|
220 shouldBe('baseOnlyDescriptor.value', derived.baseOnly); |
|
221 shouldBe('baseOnlyDescriptor.configurable', true); |
|
222 shouldBe('baseOnlyDescriptor.enumerable', false); |
|
223 shouldBe('Object.getOwnPropertyDescriptor(derived, "protoOnly")', undefined); |
|
224 var protoDupDescriptor = Object.getOwnPropertyDescriptor(derived, "protoDup"); |
|
225 shouldBe('typeof protoDupDescriptor', "object"); |
|
226 shouldBe('protoDupDescriptor.value', derived.protoDup); |
|
227 shouldBe('protoDupDescriptor.configurable', true); |
|
228 shouldBe('protoDupDescriptor.enumerable', false); |
|
229 var derivedOnlyDescriptor = Object.getOwnPropertyDescriptor(derived, "derivedOnly"); |
|
230 shouldBe('typeof derivedOnlyDescriptor', "object"); |
|
231 shouldBe('derivedOnlyDescriptor.value', derived.derivedOnly); |
|
232 shouldBe('derivedOnlyDescriptor.configurable', true); |
|
233 shouldBe('derivedOnlyDescriptor.enumerable', false); |
|
234 |
|
235 shouldBe("undefined instanceof MyObject", false); |
|
236 EvilExceptionObject.hasInstance = function f() { return f(); }; |
|
237 EvilExceptionObject.__proto__ = undefined; |
|
238 shouldThrow("undefined instanceof EvilExceptionObject"); |
|
239 EvilExceptionObject.hasInstance = function () { return true; }; |
|
240 shouldBe("undefined instanceof EvilExceptionObject", true); |
|
241 |
|
242 EvilExceptionObject.toNumber = function f() { return f(); } |
|
243 shouldThrow("EvilExceptionObject*5"); |
|
244 EvilExceptionObject.toStringExplicit = function f() { return f(); } |
|
245 shouldThrow("String(EvilExceptionObject)"); |
|
246 |
|
247 shouldBe("EmptyObject", "[object CallbackObject]"); |
|
248 |
|
249 if (failed) |
|
250 throw "Some tests failed"; |
|
251 |