1 /*
2 Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies)
3
4 This library is free software; you can redistribute it and/or
5 modify it under the terms of the GNU Library General Public
6 License as published by the Free Software Foundation; either
7 version 2 of the License, or (at your option) any later version.
8
9 This library is distributed in the hope that it will be useful,
10 but WITHOUT ANY WARRANTY; without even the implied warranty of
11 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 Library General Public License for more details.
13
14 You should have received a copy of the GNU Library General Public License
15 along with this library; see the file COPYING.LIB. If not, write to
16 the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
17 Boston, MA 02110-1301, USA.
18 */
19
20 #include "qscriptengine.h"
21 #include "qscriptprogram.h"
22 #include "qscriptsyntaxcheckresult.h"
23 #include "qscriptvalue.h"
24 #include <QtCore/qnumeric.h>
25 #include <QtTest/qtest.h>
26
27 class tst_QScriptEngine : public QObject {
28 Q_OBJECT
29
30 public:
tst_QScriptEngine()31 tst_QScriptEngine() {}
~tst_QScriptEngine()32 virtual ~tst_QScriptEngine() {}
33
34 public slots:
init()35 void init() {}
cleanup()36 void cleanup() {}
37
38 private slots:
39 void newFunction();
40 void newObject();
41 void globalObject();
42 void evaluate();
43 void collectGarbage();
44 void reportAdditionalMemoryCost();
45 void nullValue();
46 void undefinedValue();
47 void evaluateProgram();
48 void checkSyntax_data();
49 void checkSyntax();
50 void toObject();
51 void toObjectTwoEngines();
52 void newArray();
53 void uncaughtException();
54 void newDate();
55 };
56
57 /* Evaluating a script that throw an unhandled exception should return an invalid value. */
evaluate()58 void tst_QScriptEngine::evaluate()
59 {
60 QScriptEngine engine;
61 QVERIFY2(engine.evaluate("1+1").isValid(), "the expression should be evaluated and an valid result should be returned");
62 QVERIFY2(engine.evaluate("ping").isValid(), "Script throwing an unhandled exception should return an exception value");
63 }
64
myFunction(QScriptContext *,QScriptEngine * eng)65 static QScriptValue myFunction(QScriptContext*, QScriptEngine* eng)
66 {
67 return eng->nullValue();
68 }
69
myFunctionWithArg(QScriptContext *,QScriptEngine * eng,void * arg)70 static QScriptValue myFunctionWithArg(QScriptContext*, QScriptEngine* eng, void* arg)
71 {
72 int* result = reinterpret_cast<int*>(arg);
73 return QScriptValue(eng, *result);
74 }
75
myFunctionThatReturns(QScriptContext *,QScriptEngine * eng)76 static QScriptValue myFunctionThatReturns(QScriptContext*, QScriptEngine* eng)
77 {
78 return QScriptValue(eng, 42);
79 }
80
myFunctionThatReturnsWithoutEngine(QScriptContext *,QScriptEngine *)81 static QScriptValue myFunctionThatReturnsWithoutEngine(QScriptContext*, QScriptEngine*)
82 {
83 return QScriptValue(1024);
84 }
85
myFunctionThatReturnsWrongEngine(QScriptContext *,QScriptEngine *,void * arg)86 static QScriptValue myFunctionThatReturnsWrongEngine(QScriptContext*, QScriptEngine*, void* arg)
87 {
88 QScriptEngine* wrongEngine = reinterpret_cast<QScriptEngine*>(arg);
89 return QScriptValue(wrongEngine, 42);
90 }
91
newFunction()92 void tst_QScriptEngine::newFunction()
93 {
94 QScriptEngine eng;
95 {
96 QScriptValue fun = eng.newFunction(myFunction);
97 QCOMPARE(fun.isValid(), true);
98 QCOMPARE(fun.isFunction(), true);
99 QCOMPARE(fun.isObject(), true);
100 // QCOMPARE(fun.scriptClass(), (QScriptClass*)0);
101 // a prototype property is automatically constructed
102 {
103 QScriptValue prot = fun.property("prototype", QScriptValue::ResolveLocal);
104 QVERIFY(prot.isObject());
105 QVERIFY(prot.property("constructor").strictlyEquals(fun));
106 QEXPECT_FAIL("", "JSCallbackObject::getOwnPropertyDescriptor() doesn't return correct information yet", Continue);
107 QCOMPARE(fun.propertyFlags("prototype"), QScriptValue::Undeletable);
108 QEXPECT_FAIL("", "WebKit bug: 40613 (The JSObjectSetProperty doesn't overwrite property flags)", Continue);
109 QCOMPARE(prot.propertyFlags("constructor"), QScriptValue::PropertyFlags(QScriptValue::Undeletable | QScriptValue::SkipInEnumeration));
110 }
111 // prototype should be Function.prototype
112 QCOMPARE(fun.prototype().isValid(), true);
113 QCOMPARE(fun.prototype().isFunction(), true);
114 QCOMPARE(fun.prototype().strictlyEquals(eng.evaluate("Function.prototype")), true);
115
116 QCOMPARE(fun.call().isNull(), true);
117 // QCOMPARE(fun.construct().isObject(), true);
118 }
119 // the overload that takes an extra argument
120 {
121 int expectedResult = 42;
122 QScriptValue fun = eng.newFunction(myFunctionWithArg, reinterpret_cast<void*>(&expectedResult));
123 QVERIFY(fun.isFunction());
124 // QCOMPARE(fun.scriptClass(), (QScriptClass*)0);
125 // a prototype property is automatically constructed
126 {
127 QScriptValue prot = fun.property("prototype", QScriptValue::ResolveLocal);
128 QVERIFY(prot.isObject());
129 QVERIFY(prot.property("constructor").strictlyEquals(fun));
130 QEXPECT_FAIL("", "JSCallbackObject::getOwnPropertyDescriptor() doesn't return correct information yet", Continue);
131 QCOMPARE(fun.propertyFlags("prototype"), QScriptValue::Undeletable);
132 QEXPECT_FAIL("", "WebKit bug: 40613 (The JSObjectSetProperty doesn't overwrite property flags)", Continue);
133 QCOMPARE(prot.propertyFlags("constructor"), QScriptValue::PropertyFlags(QScriptValue::Undeletable | QScriptValue::SkipInEnumeration));
134 }
135 // prototype should be Function.prototype
136 QCOMPARE(fun.prototype().isValid(), true);
137 QCOMPARE(fun.prototype().isFunction(), true);
138 QCOMPARE(fun.prototype().strictlyEquals(eng.evaluate("Function.prototype")), true);
139
140 QScriptValue result = fun.call();
141 QCOMPARE(result.isNumber(), true);
142 QCOMPARE(result.toInt32(), expectedResult);
143 }
144 // the overload that takes a prototype
145 {
146 QScriptValue proto = eng.newObject();
147 QScriptValue fun = eng.newFunction(myFunction, proto);
148 QCOMPARE(fun.isValid(), true);
149 QCOMPARE(fun.isFunction(), true);
150 QCOMPARE(fun.isObject(), true);
151 // internal prototype should be Function.prototype
152 QCOMPARE(fun.prototype().isValid(), true);
153 QCOMPARE(fun.prototype().isFunction(), true);
154 QCOMPARE(fun.prototype().strictlyEquals(eng.evaluate("Function.prototype")), true);
155 // public prototype should be the one we passed
156 QCOMPARE(fun.property("prototype").strictlyEquals(proto), true);
157 QEXPECT_FAIL("", "JSCallbackObject::getOwnPropertyDescriptor() doesn't return correct information yet", Continue);
158 QCOMPARE(fun.propertyFlags("prototype"), QScriptValue::Undeletable);
159 QCOMPARE(proto.property("constructor").strictlyEquals(fun), true);
160 QEXPECT_FAIL("", "WebKit bug: 40613 (The JSObjectSetProperty doesn't overwrite property flags)", Continue);
161 QCOMPARE(proto.propertyFlags("constructor"), QScriptValue::PropertyFlags(QScriptValue::Undeletable | QScriptValue::SkipInEnumeration));
162
163 QCOMPARE(fun.call().isNull(), true);
164 // QCOMPARE(fun.construct().isObject(), true);
165 }
166 // whether the return value is correct
167 {
168 QScriptValue fun = eng.newFunction(myFunctionThatReturns);
169 QCOMPARE(fun.isValid(), true);
170 QCOMPARE(fun.isFunction(), true);
171 QCOMPARE(fun.isObject(), true);
172
173 QScriptValue result = fun.call();
174 QCOMPARE(result.isNumber(), true);
175 QCOMPARE(result.toInt32(), 42);
176 }
177 // whether the return value is assigned to the correct engine
178 {
179 QScriptValue fun = eng.newFunction(myFunctionThatReturnsWithoutEngine);
180 QCOMPARE(fun.isValid(), true);
181 QCOMPARE(fun.isFunction(), true);
182 QCOMPARE(fun.isObject(), true);
183
184 QScriptValue result = fun.call();
185 QCOMPARE(result.engine(), &eng);
186 QCOMPARE(result.isNumber(), true);
187 QCOMPARE(result.toInt32(), 1024);
188 }
189 // whether the return value is undefined when returning a value with wrong engine
190 {
191 QScriptEngine wrongEngine;
192
193 QScriptValue fun = eng.newFunction(myFunctionThatReturnsWrongEngine, reinterpret_cast<void*>(&wrongEngine));
194 QCOMPARE(fun.isValid(), true);
195 QCOMPARE(fun.isFunction(), true);
196 QCOMPARE(fun.isObject(), true);
197
198 QTest::ignoreMessage(QtWarningMsg, "Value from different engine returned from native function, returning undefined value instead.");
199 QScriptValue result = fun.call();
200 QCOMPARE(result.isValid(), true);
201 QCOMPARE(result.isUndefined(), true);
202 }
203 }
204
newObject()205 void tst_QScriptEngine::newObject()
206 {
207 QScriptEngine engine;
208 QScriptValue object = engine.newObject();
209 QVERIFY(object.isObject());
210 QVERIFY(object.engine() == &engine);
211 QVERIFY(!object.isError());
212 QVERIFY(!object.equals(engine.newObject()));
213 QVERIFY(!object.strictlyEquals(engine.newObject()));
214 QCOMPARE(object.toString(), QString::fromAscii("[object Object]"));
215 }
216
globalObject()217 void tst_QScriptEngine::globalObject()
218 {
219 QScriptEngine engine;
220 QScriptValue global = engine.globalObject();
221 QScriptValue self = engine.evaluate("this");
222 QVERIFY(global.isObject());
223 QVERIFY(engine.globalObject().equals(engine.evaluate("this")));
224 QVERIFY(engine.globalObject().strictlyEquals(self));
225 }
226
227 /* Test garbage collection, at least try to not crash. */
collectGarbage()228 void tst_QScriptEngine::collectGarbage()
229 {
230 QScriptEngine engine;
231 QScriptValue foo = engine.evaluate("( function foo() {return 'pong';} )");
232 QVERIFY(foo.isFunction());
233 engine.collectGarbage();
234 QCOMPARE(foo.call().toString(), QString::fromAscii("pong"));
235 }
236
reportAdditionalMemoryCost()237 void tst_QScriptEngine::reportAdditionalMemoryCost()
238 {
239 // There isn't any easy way to test the responsiveness of the GC;
240 // just try to call the function a few times with various sizes.
241 QScriptEngine eng;
242 for (int i = 0; i < 100; ++i) {
243 eng.reportAdditionalMemoryCost(0);
244 eng.reportAdditionalMemoryCost(10);
245 eng.reportAdditionalMemoryCost(1000);
246 eng.reportAdditionalMemoryCost(10000);
247 eng.reportAdditionalMemoryCost(100000);
248 eng.reportAdditionalMemoryCost(1000000);
249 eng.reportAdditionalMemoryCost(10000000);
250 eng.reportAdditionalMemoryCost(-1);
251 eng.reportAdditionalMemoryCost(-1000);
252 QScriptValue obj = eng.evaluate("new Object");
253 eng.collectGarbage();
254 }
255 }
256
nullValue()257 void tst_QScriptEngine::nullValue()
258 {
259 QScriptEngine engine;
260 QScriptValue value = engine.nullValue();
261 QVERIFY(value.isValid());
262 QVERIFY(value.isNull());
263 }
264
undefinedValue()265 void tst_QScriptEngine::undefinedValue()
266 {
267 QScriptEngine engine;
268 QScriptValue value = engine.undefinedValue();
269 QVERIFY(value.isValid());
270 QVERIFY(value.isUndefined());
271 }
272
evaluateProgram()273 void tst_QScriptEngine::evaluateProgram()
274 {
275 QScriptEngine eng;
276 {
277 QString code("1 + 2");
278 QString fileName("hello.js");
279 int lineNumber = 123;
280 QScriptProgram program(code, fileName, lineNumber);
281 QVERIFY(!program.isNull());
282 QCOMPARE(program.sourceCode(), code);
283 QCOMPARE(program.fileName(), fileName);
284 QCOMPARE(program.firstLineNumber(), lineNumber);
285
286 QScriptValue expected = eng.evaluate(code);
287 for (int x = 0; x < 10; ++x) {
288 QScriptValue ret = eng.evaluate(program);
289 QVERIFY(ret.equals(expected));
290 }
291
292 // operator=
293 QScriptProgram sameProgram = program;
294 QVERIFY(sameProgram == program);
295 QVERIFY(eng.evaluate(sameProgram).equals(expected));
296
297 // copy constructor
298 QScriptProgram sameProgram2(program);
299 QVERIFY(sameProgram2 == program);
300 QVERIFY(eng.evaluate(sameProgram2).equals(expected));
301
302 QScriptProgram differentProgram("2 + 3");
303 QVERIFY(differentProgram != program);
304 QVERIFY(!eng.evaluate(differentProgram).equals(expected));
305 }
306
307 // Program that accesses variable in the scope
308 {
309 QScriptProgram program("a");
310 QVERIFY(!program.isNull());
311 {
312 QScriptValue ret = eng.evaluate(program);
313 QVERIFY(ret.isError());
314 QCOMPARE(ret.toString(), QString::fromLatin1("ReferenceError: Can't find variable: a"));
315 }
316 {
317 QScriptValue ret = eng.evaluate(program);
318 QVERIFY(ret.isError());
319 }
320 eng.evaluate("a = 456");
321 {
322 QScriptValue ret = eng.evaluate(program);
323 QVERIFY(!ret.isError());
324 QCOMPARE(ret.toNumber(), 456.0);
325 }
326 }
327
328 // Program that creates closure
329 {
330 QScriptProgram program("(function() { var count = 0; return function() { return count++; }; })");
331 QVERIFY(!program.isNull());
332 QScriptValue createCounter = eng.evaluate(program);
333 QVERIFY(createCounter.isFunction());
334 QScriptValue counter = createCounter.call();
335 QVERIFY(counter.isFunction());
336 {
337 QScriptValue ret = counter.call();
338 QVERIFY(ret.isNumber());
339 }
340 QScriptValue counter2 = createCounter.call();
341 QVERIFY(counter2.isFunction());
342 QVERIFY(!counter2.equals(counter));
343 {
344 QScriptValue ret = counter2.call();
345 QVERIFY(ret.isNumber());
346 }
347 }
348
349 // Same program run in different engines
350 {
351 QString code("1 + 2");
352 QScriptProgram program(code);
353 QVERIFY(!program.isNull());
354 double expected = eng.evaluate(program).toNumber();
355 for (int x = 0; x < 2; ++x) {
356 QScriptEngine eng2;
357 for (int y = 0; y < 2; ++y) {
358 double ret = eng2.evaluate(program).toNumber();
359 QCOMPARE(ret, expected);
360 }
361 }
362 }
363
364 // No program
365 {
366 QScriptProgram program;
367 QVERIFY(program.isNull());
368 QScriptValue ret = eng.evaluate(program);
369 QVERIFY(!ret.isValid());
370 }
371 }
372
checkSyntax_data()373 void tst_QScriptEngine::checkSyntax_data()
374 {
375 QTest::addColumn<QString>("code");
376 QTest::addColumn<int>("expectedState");
377 QTest::addColumn<int>("errorLineNumber");
378 QTest::addColumn<int>("errorColumnNumber");
379 QTest::addColumn<QString>("errorMessage");
380
381 QTest::newRow("0")
382 << QString("0") << int(QScriptSyntaxCheckResult::Valid)
383 << -1 << -1 << "";
384 QTest::newRow("if (")
385 << QString("if (\n") << int(QScriptSyntaxCheckResult::Intermediate)
386 << 1 << 4 << "";
387 QTest::newRow("if else")
388 << QString("\nif else") << int(QScriptSyntaxCheckResult::Error)
389 << 2 << 4 << "SyntaxError: Parse error";
390 QTest::newRow("{if}")
391 << QString("{\n{\nif\n}\n") << int(QScriptSyntaxCheckResult::Error)
392 << 4 << 1 << "SyntaxError: Parse error";
393 QTest::newRow("foo[")
394 << QString("foo[") << int(QScriptSyntaxCheckResult::Error)
395 << 1 << 4 << "SyntaxError: Parse error";
396 QTest::newRow("foo['bar']")
397 << QString("foo['bar']") << int(QScriptSyntaxCheckResult::Valid)
398 << -1 << -1 << "";
399
400 QTest::newRow("/*")
401 << QString("/*") << int(QScriptSyntaxCheckResult::Intermediate)
402 << 1 << 1 << "Unclosed comment at end of file";
403 QTest::newRow("/*\nMy comment")
404 << QString("/*\nMy comment") << int(QScriptSyntaxCheckResult::Intermediate)
405 << 1 << 1 << "Unclosed comment at end of file";
406 QTest::newRow("/*\nMy comment */\nfoo = 10")
407 << QString("/*\nMy comment */\nfoo = 10") << int(QScriptSyntaxCheckResult::Valid)
408 << -1 << -1 << "";
409 QTest::newRow("foo = 10 /*")
410 << QString("foo = 10 /*") << int(QScriptSyntaxCheckResult::Intermediate)
411 << -1 << -1 << "";
412 QTest::newRow("foo = 10; /*")
413 << QString("foo = 10; /*") << int(QScriptSyntaxCheckResult::Intermediate)
414 << 1 << 11 << "Expected `end of file'";
415 QTest::newRow("foo = 10 /* My comment */")
416 << QString("foo = 10 /* My comment */") << int(QScriptSyntaxCheckResult::Valid)
417 << -1 << -1 << "";
418
419 QTest::newRow("/=/")
420 << QString("/=/") << int(QScriptSyntaxCheckResult::Valid) << -1 << -1 << "";
421 QTest::newRow("/=/g")
422 << QString("/=/g") << int(QScriptSyntaxCheckResult::Valid) << -1 << -1 << "";
423 QTest::newRow("/a/")
424 << QString("/a/") << int(QScriptSyntaxCheckResult::Valid) << -1 << -1 << "";
425 QTest::newRow("/a/g")
426 << QString("/a/g") << int(QScriptSyntaxCheckResult::Valid) << -1 << -1 << "";
427 }
428
checkSyntax()429 void tst_QScriptEngine::checkSyntax()
430 {
431 QFETCH(QString, code);
432 QFETCH(int, expectedState);
433 QFETCH(int, errorLineNumber);
434 QFETCH(int, errorColumnNumber);
435 QFETCH(QString, errorMessage);
436
437 QScriptSyntaxCheckResult result = QScriptEngine::checkSyntax(code);
438
439 // assignment
440 {
441 QScriptSyntaxCheckResult copy = result;
442 QCOMPARE(copy.state(), result.state());
443 QCOMPARE(copy.errorLineNumber(), result.errorLineNumber());
444 QCOMPARE(copy.errorColumnNumber(), result.errorColumnNumber());
445 QCOMPARE(copy.errorMessage(), result.errorMessage());
446 }
447 {
448 QScriptSyntaxCheckResult copy(result);
449 QCOMPARE(copy.state(), result.state());
450 QCOMPARE(copy.errorLineNumber(), result.errorLineNumber());
451 QCOMPARE(copy.errorColumnNumber(), result.errorColumnNumber());
452 QCOMPARE(copy.errorMessage(), result.errorMessage());
453 }
454
455 if (expectedState == QScriptSyntaxCheckResult::Intermediate)
456 QEXPECT_FAIL("", "QScriptSyntaxCheckResult::state() doesn't return the Intermediate state", Abort);
457 QCOMPARE(result.state(), QScriptSyntaxCheckResult::State(expectedState));
458 QCOMPARE(result.errorLineNumber(), errorLineNumber);
459 if (expectedState != QScriptSyntaxCheckResult::Valid && errorColumnNumber != 1)
460 QEXPECT_FAIL("", "QScriptSyntaxCheckResult::errorColumnNumber() doesn't return correct value", Continue);
461 QCOMPARE(result.errorColumnNumber(), errorColumnNumber);
462 QCOMPARE(result.errorMessage(), errorMessage);
463 }
464
toObject()465 void tst_QScriptEngine::toObject()
466 {
467 QScriptEngine eng;
468 QVERIFY(!eng.toObject(eng.undefinedValue()).isValid());
469 QVERIFY(!eng.toObject(eng.nullValue()).isValid());
470 QVERIFY(!eng.toObject(QScriptValue()).isValid());
471
472 QScriptValue falskt(false);
473 {
474 QScriptValue tmp = eng.toObject(falskt);
475 QVERIFY(tmp.isObject());
476 QVERIFY(!falskt.isObject());
477 QVERIFY(!falskt.engine());
478 QCOMPARE(tmp.toNumber(), falskt.toNumber());
479 }
480
481 QScriptValue sant(true);
482 {
483 QScriptValue tmp = eng.toObject(sant);
484 QVERIFY(tmp.isObject());
485 QVERIFY(!sant.isObject());
486 QVERIFY(!sant.engine());
487 QCOMPARE(tmp.toNumber(), sant.toNumber());
488 }
489
490 QScriptValue number(123.0);
491 {
492 QScriptValue tmp = eng.toObject(number);
493 QVERIFY(tmp.isObject());
494 QVERIFY(!number.isObject());
495 QVERIFY(!number.engine());
496 QCOMPARE(tmp.toNumber(), number.toNumber());
497 }
498
499 QScriptValue str = QScriptValue(&eng, QString("ciao"));
500 {
501 QScriptValue tmp = eng.toObject(str);
502 QVERIFY(tmp.isObject());
503 QVERIFY(!str.isObject());
504 QCOMPARE(tmp.toString(), str.toString());
505 }
506
507 QScriptValue object = eng.evaluate("new Object");
508 {
509 QScriptValue tmp = eng.toObject(object);
510 QVERIFY(tmp.isObject());
511 QVERIFY(object.isObject());
512 QVERIFY(tmp.strictlyEquals(object));
513 }
514 }
515
toObjectTwoEngines()516 void tst_QScriptEngine::toObjectTwoEngines()
517 {
518 QScriptEngine engine1;
519 QScriptEngine engine2;
520
521 {
522 QScriptValue null = engine1.nullValue();
523 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
524 QVERIFY(!engine2.toObject(null).isValid());
525 QVERIFY(null.isValid());
526 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
527 QVERIFY(engine2.toObject(null).engine() != &engine2);
528 }
529 {
530 QScriptValue undefined = engine1.undefinedValue();
531 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
532 QVERIFY(!engine2.toObject(undefined).isValid());
533 QVERIFY(undefined.isValid());
534 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
535 QVERIFY(engine2.toObject(undefined).engine() != &engine2);
536 }
537 {
538 QScriptValue value = engine1.evaluate("1");
539 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
540 QVERIFY(engine2.toObject(value).engine() != &engine2);
541 QVERIFY(!value.isObject());
542 }
543 {
544 QScriptValue string = engine1.evaluate("'Qt'");
545 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
546 QVERIFY(engine2.toObject(string).engine() != &engine2);
547 QVERIFY(!string.isObject());
548 }
549 {
550 QScriptValue object = engine1.evaluate("new Object");
551 QTest::ignoreMessage(QtWarningMsg, "QScriptEngine::toObject: cannot convert value created in a different engine");
552 QVERIFY(engine2.toObject(object).engine() != &engine2);
553 QVERIFY(object.isObject());
554 }
555 }
556
newArray()557 void tst_QScriptEngine::newArray()
558 {
559 QScriptEngine eng;
560 QScriptValue array = eng.newArray();
561 QCOMPARE(array.isValid(), true);
562 QCOMPARE(array.isArray(), true);
563 QCOMPARE(array.isObject(), true);
564 QVERIFY(!array.isFunction());
565 // QCOMPARE(array.scriptClass(), (QScriptClass*)0);
566
567 // Prototype should be Array.prototype.
568 QCOMPARE(array.prototype().isValid(), true);
569 QCOMPARE(array.prototype().isArray(), true);
570 QCOMPARE(array.prototype().strictlyEquals(eng.evaluate("Array.prototype")), true);
571
572 QScriptValue arrayWithSize = eng.newArray(42);
573 QCOMPARE(arrayWithSize.isValid(), true);
574 QCOMPARE(arrayWithSize.isArray(), true);
575 QCOMPARE(arrayWithSize.isObject(), true);
576 QCOMPARE(arrayWithSize.property("length").toInt32(), 42);
577
578 // task 218092
579 {
580 QScriptValue ret = eng.evaluate("[].splice(0, 0, 'a')");
581 QVERIFY(ret.isArray());
582 QCOMPARE(ret.property("length").toInt32(), 0);
583 }
584 {
585 QScriptValue ret = eng.evaluate("['a'].splice(0, 1, 'b')");
586 QVERIFY(ret.isArray());
587 QCOMPARE(ret.property("length").toInt32(), 1);
588 }
589 {
590 QScriptValue ret = eng.evaluate("['a', 'b'].splice(0, 1, 'c')");
591 QVERIFY(ret.isArray());
592 QCOMPARE(ret.property("length").toInt32(), 1);
593 }
594 {
595 QScriptValue ret = eng.evaluate("['a', 'b', 'c'].splice(0, 2, 'd')");
596 QVERIFY(ret.isArray());
597 QCOMPARE(ret.property("length").toInt32(), 2);
598 }
599 {
600 QScriptValue ret = eng.evaluate("['a', 'b', 'c'].splice(1, 2, 'd', 'e', 'f')");
601 QVERIFY(ret.isArray());
602 QCOMPARE(ret.property("length").toInt32(), 2);
603 }
604 }
605
uncaughtException()606 void tst_QScriptEngine::uncaughtException()
607 {
608 QScriptEngine eng;
609 QScriptValue fun = eng.evaluate("(function foo () { return null; });");
610 QVERIFY(!eng.uncaughtException().isValid());
611 QVERIFY(fun.isFunction());
612 QScriptValue throwFun = eng.evaluate("( function() { throw new Error('Pong'); });");
613 QVERIFY(throwFun.isFunction());
614 {
615 eng.evaluate("a = 10");
616 QVERIFY(!eng.hasUncaughtException());
617 QVERIFY(!eng.uncaughtException().isValid());
618 }
619 {
620 eng.evaluate("1 = 2");
621 QVERIFY(eng.hasUncaughtException());
622 eng.clearExceptions();
623 QVERIFY(!eng.hasUncaughtException());
624 }
625 {
626 // Check if the call or toString functions can remove the last exception.
627 QVERIFY(throwFun.call().isError());
628 QVERIFY(eng.hasUncaughtException());
629 QScriptValue exception = eng.uncaughtException();
630 fun.call();
631 exception.toString();
632 QVERIFY(eng.hasUncaughtException());
633 QVERIFY(eng.uncaughtException().strictlyEquals(exception));
634 }
635 eng.clearExceptions();
636 {
637 // Check if in the call function a new exception can override an existing one.
638 throwFun.call();
639 QVERIFY(eng.hasUncaughtException());
640 QScriptValue exception = eng.uncaughtException();
641 throwFun.call();
642 QVERIFY(eng.hasUncaughtException());
643 QVERIFY(!exception.strictlyEquals(eng.uncaughtException()));
644 }
645 {
646 eng.evaluate("throwFun = (function foo () { throw new Error('bla') });");
647 eng.evaluate("1;\nthrowFun();");
648 QVERIFY(eng.hasUncaughtException());
649 QCOMPARE(eng.uncaughtExceptionLineNumber(), 1);
650 eng.clearExceptions();
651 QVERIFY(!eng.hasUncaughtException());
652 }
653 for (int x = 1; x < 4; ++x) {
654 QScriptValue ret = eng.evaluate("a = 10;\nb = 20;\n0 = 0;\n",
655 QString::fromLatin1("FooScript") + QString::number(x),
656 /* lineNumber */ x);
657 QVERIFY(eng.hasUncaughtException());
658 QCOMPARE(eng.uncaughtExceptionLineNumber(), x + 2);
659 QVERIFY(eng.uncaughtException().strictlyEquals(ret));
660 QVERIFY(eng.hasUncaughtException());
661 QVERIFY(eng.uncaughtException().strictlyEquals(ret));
662 QString backtrace = QString::fromLatin1("<anonymous>()@FooScript") + QString::number(x) + ":" + QString::number(x + 2);
663 QCOMPARE(eng.uncaughtExceptionBacktrace().join(""), backtrace);
664 QVERIFY(fun.call().isNull());
665 QVERIFY(eng.hasUncaughtException());
666 QCOMPARE(eng.uncaughtExceptionLineNumber(), x + 2);
667 QVERIFY(eng.uncaughtException().strictlyEquals(ret));
668 eng.clearExceptions();
669 QVERIFY(!eng.hasUncaughtException());
670 QCOMPARE(eng.uncaughtExceptionLineNumber(), -1);
671 QVERIFY(!eng.uncaughtException().isValid());
672 eng.evaluate("2 = 3");
673 QVERIFY(eng.hasUncaughtException());
674 QScriptValue ret2 = throwFun.call();
675 QVERIFY(ret2.isError());
676 QVERIFY(eng.hasUncaughtException());
677 QVERIFY(eng.uncaughtException().strictlyEquals(ret2));
678 QCOMPARE(eng.uncaughtExceptionLineNumber(), 1);
679 eng.clearExceptions();
680 QVERIFY(!eng.hasUncaughtException());
681 eng.evaluate("1 + 2");
682 QVERIFY(!eng.hasUncaughtException());
683 }
684 }
685
newDate()686 void tst_QScriptEngine::newDate()
687 {
688 QScriptEngine eng;
689 {
690 QScriptValue date = eng.newDate(0);
691 QCOMPARE(date.isValid(), true);
692 QCOMPARE(date.isDate(), true);
693 QCOMPARE(date.isObject(), true);
694 QVERIFY(!date.isFunction());
695 // prototype should be Date.prototype
696 QCOMPARE(date.prototype().isValid(), true);
697 QCOMPARE(date.prototype().isDate(), true);
698 QCOMPARE(date.prototype().strictlyEquals(eng.evaluate("Date.prototype")), true);
699 }
700 {
701 QDateTime dt = QDateTime(QDate(1, 2, 3), QTime(4, 5, 6, 7), Qt::LocalTime);
702 QScriptValue date = eng.newDate(dt);
703 QCOMPARE(date.isValid(), true);
704 QCOMPARE(date.isDate(), true);
705 QCOMPARE(date.isObject(), true);
706 // prototype should be Date.prototype
707 QCOMPARE(date.prototype().isValid(), true);
708 QCOMPARE(date.prototype().isDate(), true);
709 QCOMPARE(date.prototype().strictlyEquals(eng.evaluate("Date.prototype")), true);
710
711 QCOMPARE(date.toDateTime(), dt);
712 }
713 {
714 QDateTime dt = QDateTime(QDate(1, 2, 3), QTime(4, 5, 6, 7), Qt::UTC);
715 QScriptValue date = eng.newDate(dt);
716 // toDateTime() result should be in local time
717 QCOMPARE(date.toDateTime(), dt.toLocalTime());
718 }
719 // Date.parse() should return NaN when it fails
720 {
721 QScriptValue ret = eng.evaluate("Date.parse()");
722 QVERIFY(ret.isNumber());
723 QVERIFY(qIsNaN(ret.toNumber()));
724 }
725 // Date.parse() should be able to parse the output of Date().toString()
726 {
727 QScriptValue ret = eng.evaluate("var x = new Date(); var s = x.toString(); s == new Date(Date.parse(s)).toString()");
728 QVERIFY(ret.isBoolean());
729 QCOMPARE(ret.toBoolean(), true);
730 }
731 }
732
733 QTEST_MAIN(tst_QScriptEngine)
734 #include "tst_qscriptengine.moc"
735