blob: 48393e9b092a49f0e2d8da626638f1d63ddf763f [file] [log] [blame]
// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import "package:expect/expect.dart";
// Invocation and noSuchMethod testing.
Map<Symbol, dynamic> listToNamedArguments(list) {
var iterator = list.iterator;
var result = new Map<Symbol, dynamic>();
while (iterator.moveNext()) {
Symbol key = iterator.current;
Expect.isTrue(iterator.moveNext());
result[key] = iterator.current;
}
return result;
}
/** Class with noSuchMethod that returns the mirror */
class N {
// Storage for the last argument to noSuchMethod.
// Needed for setters, which don't evaluate to the return value.
var last;
noSuchMethod(Invocation m) => last = m;
flif(int x) { Expect.fail("never get here"); }
flaf([int x]) { Expect.fail("never get here"); }
flof({int y}) { Expect.fail("never get here"); }
get wut => this;
final int plif = 99;
int get plaf { Expect.fail("never get here"); return 0; }
}
/** As [N] but also implements 'call', so we can call it with wrong arguments.*/
class C extends N {
call(int x) { Expect.fail("never get here"); }
}
/**
* Checks the data of an Invocation.
*
* Call without optionals for getters, with only positional for setters,
* and with both optionals for everything else.
*/
testInvocationMirror(Invocation im, Symbol name,
[List positional, List named]) {
Expect.isTrue(im is Invocation, "is Invocation");
Expect.equals(name, im.memberName, "name");
if (named == null) {
Expect.isTrue(im.isAccessor, "$name:isAccessor");
Expect.isFalse(im.isMethod, "$name:isMethod");
if (positional == null) {
Expect.isTrue(im.isGetter, "$name:isGetter");
Expect.isFalse(im.isSetter, "$name:isSetter");
Expect.equals(0, im.positionalArguments.length, "$name:#positional");
Expect.equals(0, im.namedArguments.length, "$name:#named");
return;
}
Expect.isTrue(im.isSetter, "$name:isSetter");
Expect.isFalse(im.isGetter, "$name:isGetter");
Expect.equals(1, im.positionalArguments.length, "$name:#positional");
Expect.equals(positional[0], im.positionalArguments[0],
"$name:positional[0]");
Expect.equals(0, im.namedArguments.length, "$name:#named");
return;
}
Map<Symbol, dynamic> namedArguments = listToNamedArguments(named);
Expect.isTrue(im.isMethod, "$name:isMethod");
Expect.isFalse(im.isAccessor, "$name:isAccessor");
Expect.isFalse(im.isSetter, "$name:isSetter");
Expect.isFalse(im.isGetter, "$name:isGetter");
Expect.equals(positional.length, im.positionalArguments.length);
for (int i = 0; i < positional.length; i++) {
Expect.equals(positional[i], im.positionalArguments[i],
"$name:positional[$i]");
}
Expect.equals(namedArguments.length, im.namedArguments.length,
"$name:#named");
namedArguments.forEach((k, v) {
Expect.isTrue(im.namedArguments.containsKey(k),
"$name:?namedArguments[$k]");
Expect.equals(v, im.namedArguments[k], "$name:namedArguments[$k]");
});
}
// Test different ways that noSuchMethod can be called.
testInvocationMirrors() {
var n = new N();
var c = new C();
// Missing property/method access.
testInvocationMirror(n.bar, const Symbol('bar'));
testInvocationMirror((n..bar = 42).last, const Symbol('bar='), [42]);
testInvocationMirror(n.bar(), const Symbol('bar'), [], []);
testInvocationMirror(n.bar(42), const Symbol('bar'), [42], []);
testInvocationMirror(n.bar(x: 42), const Symbol('bar'), [],
[const Symbol("x"), 42]);
testInvocationMirror(n.bar(37, x: 42), const Symbol('bar'), [37],
[const Symbol("x"), 42]);
// Missing operator access.
testInvocationMirror(n + 4, const Symbol('+'), [4], []);
testInvocationMirror(n - 4, const Symbol('-'), [4], []);
testInvocationMirror(-n, const Symbol('unary-'), [], []);
testInvocationMirror(n[42], const Symbol('[]'), [42], []);
testInvocationMirror((n..[37] = 42).last, const Symbol('[]='), [37, 42], []);
// Calling as function when it's not.
testInvocationMirror(n(), const Symbol('call'), [], []);
testInvocationMirror(n(42), const Symbol('call'), [42], []);
testInvocationMirror(n(x: 42), const Symbol('call'), [],
[const Symbol("x"), 42]);
testInvocationMirror(n(37, x: 42), const Symbol('call'), [37],
[const Symbol("x"), 42]);
// Calling with arguments not matching existing call method.
testInvocationMirror(c(), const Symbol('call'), [], []);
testInvocationMirror(c(37, 42), const Symbol('call'), [37, 42], []);
testInvocationMirror(c(x: 42), const Symbol('call'), [],
[const Symbol("x"), 42]);
testInvocationMirror(c(37, x: 42), const Symbol('call'), [37],
[const Symbol("x"), 42]);
// Wrong arguments to existing function.
testInvocationMirror(n.flif(), const Symbol("flif"), [], []);
testInvocationMirror(n.flif(37, 42), const Symbol("flif"), [37, 42], []);
testInvocationMirror(n.flif(x: 42), const Symbol("flif"), [],
[const Symbol("x"), 42]);
testInvocationMirror(n.flif(37, x: 42), const Symbol("flif"), [37],
[const Symbol("x"), 42]);
testInvocationMirror((n..flif = 42).last, const Symbol("flif="), [42]);
testInvocationMirror(n.flaf(37, 42), const Symbol("flaf"), [37, 42], []);
testInvocationMirror(n.flaf(x: 42), const Symbol("flaf"), [],
[const Symbol("x"), 42]);
testInvocationMirror(n.flaf(37, x: 42), const Symbol("flaf"), [37],
[const Symbol("x"), 42]);
testInvocationMirror((n..flaf = 42).last, const Symbol("flaf="), [42]);
testInvocationMirror(n.flof(37, 42), const Symbol("flof"), [37, 42], []);
testInvocationMirror(n.flof(x: 42), const Symbol("flof"), [],
[const Symbol("x"), 42]);
testInvocationMirror(n.flof(37, y: 42), const Symbol("flof"), [37],
[const Symbol("y"), 42]);
testInvocationMirror((n..flof = 42).last, const Symbol("flof="), [42]);
// Reading works.
Expect.isTrue(n.flif is Function);
Expect.isTrue(n.flaf is Function);
Expect.isTrue(n.flof is Function);
// Writing to read-only fields.
testInvocationMirror((n..wut = 42).last, const Symbol("wut="), [42]);
testInvocationMirror((n..plif = 42).last, const Symbol("plif="), [42]);
testInvocationMirror((n..plaf = 42).last, const Symbol("plaf="), [42]);
// Trick call to n.call - wut is a getter returning n again.
testInvocationMirror(n.wut(42), const Symbol("call"), [42], []);
// Calling noSuchMethod itself, badly.
testInvocationMirror(n.noSuchMethod(), const Symbol("noSuchMethod"), [], []);
testInvocationMirror(n.noSuchMethod(37, 42), const Symbol("noSuchMethod"),
[37, 42], []);
testInvocationMirror(n.noSuchMethod(37, x:42),
const Symbol("noSuchMethod"), [37],
[const Symbol("x"), 42]);
testInvocationMirror(n.noSuchMethod(x:42), const Symbol("noSuchMethod"), [],
[const Symbol("x"), 42]);
// Closurizing a method means that calling it badly will not hit the
// original receivers noSuchMethod, only the one inherited from Object
// by the closure object.
Expect.throws(() { var x = n.flif; x(37, 42); },
(e) => e is NoSuchMethodError);
Expect.throws(() { var x = c.call; x(37, 42); },
(e) => e is NoSuchMethodError);
}
class M extends N {
noSuchMethod(Invocation m) { throw "never get here"; }
testSuperCalls() {
// Missing property/method access.
testInvocationMirror(super.bar, const Symbol('bar'));
testInvocationMirror((){super.bar = 42; return last;}(),
const Symbol('bar='), [42]);
testInvocationMirror(super.bar(), const Symbol('bar'), [], []);
testInvocationMirror(super.bar(42), const Symbol('bar'), [42], []);
testInvocationMirror(super.bar(x: 42), const Symbol('bar'), [],
[const Symbol("x"), 42]);
testInvocationMirror(super.bar(37, x: 42), const Symbol('bar'), [37],
[const Symbol("x"), 42]);
// Missing operator access.
testInvocationMirror(super + 4, const Symbol('+'), [4], []);
testInvocationMirror(super - 4, const Symbol('-'), [4], []);
testInvocationMirror(-super, const Symbol('unary-'), [], []);
testInvocationMirror(super[42], const Symbol('[]'), [42], []);
testInvocationMirror((){super[37] = 42; return last;}(),
const Symbol('[]='), [37, 42], []);
// Wrong arguments to existing function.
testInvocationMirror(super.flif(), const Symbol("flif"), [], []);
testInvocationMirror(super.flif(37, 42), const Symbol("flif"), [37, 42],
[]);
testInvocationMirror(super.flif(x: 42), const Symbol("flif"), [],
[const Symbol("x"), 42]);
testInvocationMirror(super.flif(37, x: 42), const Symbol("flif"), [37],
[const Symbol("x"), 42]);
testInvocationMirror((){super.flif = 42; return last;}(),
const Symbol("flif="), [42]);
testInvocationMirror(super.flaf(37, 42), const Symbol("flaf"), [37, 42],
[]);
testInvocationMirror(super.flaf(x: 42), const Symbol("flaf"), [],
[const Symbol("x"), 42]);
testInvocationMirror(super.flaf(37, x: 42), const Symbol("flaf"), [37],
[const Symbol("x"), 42]);
testInvocationMirror((){super.flaf = 42; return last;}(),
const Symbol("flaf="), [42]);
testInvocationMirror(super.flof(37, 42), const Symbol("flof"), [37, 42],
[]);
testInvocationMirror(super.flof(x: 42), const Symbol("flof"), [],
[const Symbol("x"), 42]);
testInvocationMirror(super.flof(37, y: 42), const Symbol("flof"), [37],
[const Symbol("y"), 42]);
testInvocationMirror((){super.flof = 42; return last;}(),
const Symbol("flof="), [42]);
// Reading works.
Expect.isTrue(super.flif is Function);
Expect.isTrue(super.flaf is Function);
Expect.isTrue(super.flof is Function);
// Writing to read-only fields.
testInvocationMirror((){super.wut = 42; return last;}(),
const Symbol("wut="), [42]);
testInvocationMirror((){super.plif = 42; return last;}(),
const Symbol("plif="), [42]);
testInvocationMirror((){super.plaf = 42; return last;}(),
const Symbol("plaf="), [42]);
// Calling noSuchMethod itself, badly.
testInvocationMirror(super.noSuchMethod(), const Symbol("noSuchMethod"), [],
[]);
testInvocationMirror(super.noSuchMethod(37, 42),
const Symbol("noSuchMethod"), [37, 42], []);
testInvocationMirror(super.noSuchMethod(37, x:42),
const Symbol("noSuchMethod"), [37],
[const Symbol("x"), 42]);
testInvocationMirror(super.noSuchMethod(x:42),
const Symbol("noSuchMethod"), [],
[const Symbol("x"), 42]);
// Closurizing a method means that calling it badly will not hit the
// original receivers noSuchMethod, only the one inherited from Object
// by the closure object.
Expect.throws(() { var x = super.flif; x(37, 42); },
(e) => e is NoSuchMethodError);
}
}
// Test the NoSuchMethodError thrown by different incorrect calls.
testNoSuchMethodErrors() {
test(Function block) {
Expect.throws(block, (e) => e is NoSuchMethodError);
}
var n = new N();
var o = new Object();
test(() => o.bar);
test(() => o.bar = 42);
test(() => o.bar());
test(() => o + 2);
test(() => -o);
test(() => o[0]);
test(() => o[0] = 42);
test(() => o());
test(() => o.toString = 42);
test(() => o.toString(42));
test(() => o.toString(x: 37));
test(() => o.hashCode = 42);
test(() => o.hashCode()); // Thrown by int.noSuchMethod.
test(() => (n.flif)()); // Extracted method has no noSuchMethod.
}
main() {
testInvocationMirrors();
testNoSuchMethodErrors();
new M().testSuperCalls();
}