| // Copyright (c) 2013, 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. |
| |
| // This tests uses the multi-test "ok" feature: |
| // none: Trimmed behaviour. Passing on the VM. |
| // 01: Trimmed version for dart2js. |
| // 02: Full version passing in the VM. |
| // |
| // TODO(rmacnak,ahe): Remove multi-test when VM and dart2js are on par. |
| |
| library test.class_equality_test; |
| |
| import 'dart:mirrors'; |
| |
| import 'package:expect/expect.dart'; |
| |
| class A<T> {} |
| |
| class B extends A<int> {} |
| |
| class BadEqualityHash { |
| int count = 0; |
| bool operator ==(other) => true; |
| int get hashCode => count++; |
| } |
| |
| typedef bool Predicate(Object o); |
| Predicate somePredicate = (Object o) => false; |
| |
| checkEquality(List<Map<String, Object?>> equivalenceClasses) { |
| for (var equivalenceClass in equivalenceClasses) { |
| equivalenceClass.forEach((name, member) { |
| equivalenceClass.forEach((otherName, otherMember) { |
| // Reflexivity, symmetry and transitivity. |
| Expect.equals(member, otherMember, "$name == $otherName"); |
| Expect.equals( |
| member.hashCode, |
| otherMember.hashCode, |
| "$name.hashCode == $otherName.hashCode", |
| ); |
| }); |
| for (var otherEquivalenceClass in equivalenceClasses) { |
| if (otherEquivalenceClass == equivalenceClass) continue; |
| otherEquivalenceClass.forEach((otherName, otherMember) { |
| Expect.notEquals( |
| member, |
| otherMember, |
| "$name != $otherName", |
| ); // Exclusion. |
| // Hash codes may or may not be equal. |
| }); |
| } |
| }); |
| } |
| } |
| |
| void subroutine() {} |
| |
| main() { |
| LibraryMirror thisLibrary = currentMirrorSystem().findLibrary( |
| const Symbol('test.class_equality_test'), |
| ); |
| |
| var o1 = new Object(); |
| var o2 = new Object(); |
| |
| var badEqualityHash1 = new BadEqualityHash(); |
| var badEqualityHash2 = new BadEqualityHash(); |
| |
| checkEquality(<Map<String, Object?>>[ |
| {'reflect(o1)': reflect(o1), 'reflect(o1), again': reflect(o1)}, |
| {'reflect(o2)': reflect(o2), 'reflect(o2), again': reflect(o2)}, |
| { |
| 'reflect(badEqualityHash1)': reflect(badEqualityHash1), |
| 'reflect(badEqualityHash1), again': reflect(badEqualityHash1), |
| }, |
| { |
| 'reflect(badEqualityHash2)': reflect(badEqualityHash2), |
| 'reflect(badEqualityHash2), again': reflect(badEqualityHash2), |
| }, |
| {'reflect(true)': reflect(true), 'reflect(true), again': reflect(true)}, |
| {'reflect(false)': reflect(false), 'reflect(false), again': reflect(false)}, |
| {'reflect(null)': reflect(null), 'reflect(null), again': reflect(null)}, |
| { |
| 'reflect(3.5+4.5)': reflect(3.5 + 4.5), |
| 'reflect(6.5+1.5)': reflect(6.5 + 1.5), |
| }, |
| {'reflect(3+4)': reflect(3 + 4), 'reflect(6+1)': reflect(6 + 1)}, |
| {'reflect("foo")': reflect("foo"), 'reflect("foo"), again': reflect("foo")}, |
| { |
| 'currentMirrorSystem().voidType': currentMirrorSystem().voidType, |
| 'thisLibrary.declarations[#subroutine].returnType': |
| (thisLibrary.declarations[#subroutine] as MethodMirror).returnType, |
| }, |
| { |
| 'currentMirrorSystem().dynamicType': currentMirrorSystem().dynamicType, |
| 'thisLibrary.declarations[#main].returnType': |
| (thisLibrary.declarations[#main] as MethodMirror).returnType, |
| }, |
| { |
| 'reflectClass(A)': reflectClass(A), |
| 'thisLibrary.declarations[#A]': thisLibrary.declarations[#A], |
| 'reflect(new A<int>()).type.originalDeclaration': reflect( |
| new A<int>(), |
| ).type.originalDeclaration, |
| }, |
| { |
| 'reflectClass(B).superclass': reflectClass(B).superclass, |
| 'reflect(new A<int>()).type': reflect(new A<int>()).type, |
| }, |
| { |
| 'reflectClass(B)': reflectClass(B), |
| 'thisLibrary.declarations[#B]': thisLibrary.declarations[#B], |
| 'reflect(new B()).type': reflect(new B()).type, |
| }, |
| { |
| 'reflectClass(BadEqualityHash).declarations[#==]': reflectClass( |
| BadEqualityHash, |
| ).declarations[#==], |
| 'reflect(new BadEqualityHash()).type.declarations[#==]': reflect( |
| new BadEqualityHash(), |
| ).type.declarations[#==], |
| }, |
| { |
| 'reflectClass(BadEqualityHash).declarations[#==].parameters[0]': |
| (reflectClass(BadEqualityHash).declarations[#==] as MethodMirror) |
| .parameters[0], |
| 'reflect(new BadEqualityHash()).type.declarations[#==].parameters[0]': |
| (reflect(new BadEqualityHash()).type.declarations[#==] |
| as MethodMirror) |
| .parameters[0], |
| }, |
| { |
| 'reflectClass(BadEqualityHash).declarations[#count]': reflectClass( |
| BadEqualityHash, |
| ).declarations[#count], |
| 'reflect(new BadEqualityHash()).type.declarations[#count]': reflect( |
| new BadEqualityHash(), |
| ).type.declarations[#count], |
| }, |
| { |
| 'reflectType(Predicate)': reflectType(Predicate), |
| 'thisLibrary.declarations[#somePredicate].type': |
| (thisLibrary.declarations[#somePredicate] as VariableMirror).type, |
| }, |
| { |
| 'reflectType(Predicate).referent': |
| (reflectType(Predicate) as TypedefMirror).referent, |
| 'thisLibrary.declarations[#somePredicate].type.referent': |
| ((thisLibrary.declarations[#somePredicate] as VariableMirror).type |
| as TypedefMirror) |
| .referent, |
| }, |
| { |
| 'reflectClass(A).typeVariables.single': reflectClass( |
| A, |
| ).typeVariables.single, |
| 'reflect(new A<int>()).type.originalDeclaration.typeVariables.single': |
| reflect(new A<int>()).type.originalDeclaration.typeVariables.single, |
| }, |
| {'currentMirrorSystem()': currentMirrorSystem()}, |
| {'currentMirrorSystem().isolate': currentMirrorSystem().isolate}, |
| { |
| 'thisLibrary': thisLibrary, |
| 'reflectClass(A).owner': reflectClass(A).owner, |
| 'reflectClass(B).owner': reflectClass(B).owner, |
| 'reflect(new A()).type.owner': reflect(new A()).type.owner, |
| 'reflect(new B()).type.owner': reflect(new B()).type.owner, |
| }, |
| ]); |
| } |