blob: c6af6b408266645491d53f5f0e125dc3ed3eec79 [file] [log] [blame]
// Copyright (c) 2011, 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.
// Dart test program testing that NoSuchMethod is properly called.
import "package:expect/expect.dart";
class Cat {
bool eatFood(String food) => true;
String scratch(String furniture) => 'purr';
String mood = '';
}
class MockCat implements Cat {
dynamic noSuchMethod(Invocation invocation) {
var arg = invocation.positionalArguments[0];
return arg is String && arg.isNotEmpty;
}
}
class MockCat2 extends MockCat {
// this apparently works.
noSuchMethod(_);
}
class MockCat3 extends MockCat2 implements Cat {
bool eatFood(String food, {double amount});
String scratch(String furniture, [String? furniture2]);
dynamic noSuchMethod(Invocation invocation) {
if (invocation.memberName == #scratch) {
return invocation.positionalArguments.join(',');
}
return (invocation.positionalArguments[0] as String).isNotEmpty &&
invocation.namedArguments[#amount] > 0.5;
}
}
class MockWithGenerics {
List<Type> doStuff<T>(T t);
noSuchMethod(i) => (i as dynamic).typeArguments;
}
class MockWithGetterSetter {
get getter;
set setter(value);
late Invocation invocation;
noSuchMethod(i) {
invocation = i;
}
}
class Callable {
int call() => 1;
int m() => 2;
}
class MockCallable implements Callable {
noSuchMethod(i) => i.memberName == #call ? 42 : 0;
}
void main() {
MockCat mock = new MockCat();
Expect.isTrue((mock as dynamic).eatFood("cat food"));
Expect.isFalse(mock.eatFood(""));
mock.mood = 'sleepy';
(mock as dynamic).mood = 'playful';
Expect.throwsTypeError(() {
(mock as dynamic).mood = 42;
});
// In strong mode this will be a runtime type error:
// bool is not a String. VM will fail with noSuchMethod +.
Expect.throws(() => mock.scratch("couch") + '');
var mock2 = new MockCat2();
Expect.isTrue(mock2.eatFood("cat food"));
var mock3 = new MockCat3();
Expect.isTrue(mock3.eatFood("cat food", amount: 0.9));
Expect.isFalse(mock3.eatFood("cat food", amount: 0.3));
Expect.equals("chair,null", mock3.scratch("chair"));
Expect.equals("chair,couch", mock3.scratch("chair", "couch"));
Expect.equals("chair,null", mock3.scratch("chair", null));
Expect.equals("chair,", mock3.scratch("chair", ""));
var g = new MockWithGenerics();
Expect.listEquals([int], g.doStuff(42));
Expect.listEquals([num], g.doStuff<num>(42));
Expect.listEquals([String], g.doStuff('hi'));
var s = new MockWithGetterSetter();
s.getter;
Expect.equals(0, s.invocation.positionalArguments.length);
Expect.isTrue(s.invocation.isGetter);
Expect.isFalse(s.invocation.isSetter);
Expect.isFalse(s.invocation.isMethod);
s.setter = 42;
Expect.equals(42, s.invocation.positionalArguments.single);
Expect.isFalse(s.invocation.isGetter);
Expect.isTrue(s.invocation.isSetter);
Expect.isFalse(s.invocation.isMethod);
testMockTearoffs();
testMockCallable();
testMockCallableTearoff();
}
testMockCallable() {
Callable call = new MockCallable();
Expect.equals(42, call());
Expect.equals(42, (call as dynamic)());
Expect.equals(0, call.m());
Expect.equals(0, (call as dynamic).m());
}
testMockCallableTearoff() {
var mock = new MockCallable();
Function f = mock;
Expect.equals(42, f());
Expect.equals(42, (f as dynamic)());
Expect.equals(f, mock.call);
Expect.equals(f.call, mock.call);
Expect.equals((f as dynamic).call, mock.call);
Expect.equals(f.call, (mock as dynamic).call);
}
typedef bool EatFoodType(String food);
testMockTearoffs() {
var mock2 = new MockCat2();
var eat = mock2.eatFood;
var eat2 = (mock2 as dynamic).eatFood;
Expect.isTrue(eat is EatFoodType, 'eat is EatFoodType');
Expect.isTrue(eat2 is EatFoodType, 'eat2 is EatFoodType');
Expect.equals(eat, eat2, 'eat == eat2');
Expect.isTrue(eat.runtimeType == eat2.runtimeType,
'eat.runtimeType == eat2.runtimeType');
Expect.isTrue(eat("cat food"), 'eat("cat food")');
Expect.isFalse(eat(""), 'eat("")');
Expect.isTrue(eat2("cat food"), 'eat2("cat food")');
Expect.isFalse(eat2(""), 'eat2("")');
var g = new MockWithGenerics();
var doStuff = g.doStuff;
var doStuff2 = (g as dynamic).doStuff;
Expect.equals(doStuff, doStuff2, 'doStuff == doStuff2');
Expect.equals(doStuff.runtimeType, doStuff2.runtimeType,
'doStuff.runtimeType == doStuff2.runtimeType');
Expect.listEquals([int], doStuff(42));
Expect.listEquals([num], doStuff<num>(42));
Expect.listEquals([String], doStuff('hi'));
// no inference happens because `doStuff2` is dynamic.
Expect.listEquals([num], doStuff2<num>(42));
expectIsDynamicOrObject(List types) {
Expect.equals(1, types.length);
var t = types[0];
// TODO(jmesserly): allows either type because of
// https://github.com/dart-lang/sdk/issues/32483
Expect.isTrue(t == dynamic || t == Object, '$t == dynamic || $t == Object');
}
expectIsDynamicOrObject(doStuff2(42));
expectIsDynamicOrObject(doStuff2('hi'));
}