|  | // 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. | 
|  |  | 
|  | /// Helper methods for converting a [Mirror] to a [String]. | 
|  | library test.stringify; | 
|  |  | 
|  | import 'dart:mirrors'; | 
|  |  | 
|  | import 'package:expect/expect.dart'; | 
|  |  | 
|  | String name(DeclarationMirror? mirror) { | 
|  | return (mirror == null) ? '<null>' : stringify(mirror.simpleName); | 
|  | } | 
|  |  | 
|  | String stringifyMap(Map map) { | 
|  | var buffer = new StringBuffer(); | 
|  | bool first = true; | 
|  | var names = map.keys.map((s) => MirrorSystem.getName(s)).toList()..sort(); | 
|  | for (String key in names) { | 
|  | if (!first) buffer.write(', '); | 
|  | first = false; | 
|  | buffer.write(key); | 
|  | buffer.write(': '); | 
|  | buffer.write(stringify(map[new Symbol(key)])); | 
|  | } | 
|  | return '{$buffer}'; | 
|  | } | 
|  |  | 
|  | String stringifyIterable(Iterable list) { | 
|  | var buffer = new StringBuffer(); | 
|  | bool first = true; | 
|  | for (String value in list.map(stringify)) { | 
|  | if (!first) buffer.write(', '); | 
|  | first = false; | 
|  | buffer.write(value); | 
|  | } | 
|  | return '[$buffer]'; | 
|  | } | 
|  |  | 
|  | String stringifyInstance(InstanceMirror instance) { | 
|  | var buffer = new StringBuffer(); | 
|  | if (instance.hasReflectee) { | 
|  | buffer.write('value = ${stringify(instance.reflectee)}'); | 
|  | } | 
|  | return 'Instance(${buffer})'; | 
|  | } | 
|  |  | 
|  | String stringifySymbol(Symbol symbol) => 's(${MirrorSystem.getName(symbol)})'; | 
|  |  | 
|  | void writeDeclarationOn(DeclarationMirror mirror, StringBuffer buffer) { | 
|  | buffer.write(stringify(mirror.simpleName)); | 
|  | if (mirror.owner != null) { | 
|  | buffer.write(' in '); | 
|  | buffer.write(name(mirror.owner)); | 
|  | } | 
|  | if (mirror.isPrivate) buffer.write(', private'); | 
|  | if (mirror.isTopLevel) buffer.write(', top-level'); | 
|  | } | 
|  |  | 
|  | void writeVariableOn(VariableMirror variable, StringBuffer buffer) { | 
|  | writeDeclarationOn(variable, buffer); | 
|  | if (variable.isStatic) buffer.write(', static'); | 
|  | if (variable.isFinal) buffer.write(', final'); | 
|  | if (variable.isConst) buffer.write(', const'); | 
|  | } | 
|  |  | 
|  | String stringifyVariable(VariableMirror variable) { | 
|  | var buffer = new StringBuffer(); | 
|  | writeVariableOn(variable, buffer); | 
|  | return 'Variable($buffer)'; | 
|  | } | 
|  |  | 
|  | String stringifyParameter(ParameterMirror parameter) { | 
|  | var buffer = new StringBuffer(); | 
|  | writeVariableOn(parameter, buffer); | 
|  | if (parameter.isOptional) buffer.write(', optional'); | 
|  | if (parameter.isNamed) buffer.write(', named'); | 
|  | // TODO(6490): dart2js always returns false for hasDefaultValue. | 
|  | if (parameter.hasDefaultValue) { | 
|  | buffer.write(', value = ${stringify(parameter.defaultValue)}'); | 
|  | } | 
|  | // TODO(ahe): Move to writeVariableOn. | 
|  | buffer.write(', type = ${stringify(parameter.type)}'); | 
|  | return 'Parameter($buffer)'; | 
|  | } | 
|  |  | 
|  | String stringifyTypeVariable(TypeVariableMirror typeVariable) { | 
|  | var buffer = new StringBuffer(); | 
|  | writeDeclarationOn(typeVariable, buffer); | 
|  | buffer.write(', upperBound = ${stringify(typeVariable.upperBound)}'); | 
|  | return 'TypeVariable($buffer)'; | 
|  | } | 
|  |  | 
|  | String stringifyType(TypeMirror type) { | 
|  | var buffer = new StringBuffer(); | 
|  | writeDeclarationOn(type, buffer); | 
|  | return 'Type($buffer)'; | 
|  | } | 
|  |  | 
|  | String stringifyClass(ClassMirror cls) { | 
|  | var buffer = new StringBuffer(); | 
|  | writeDeclarationOn(cls, buffer); | 
|  | return 'Class($buffer)'; | 
|  | } | 
|  |  | 
|  | String stringifyMethod(MethodMirror method) { | 
|  | var buffer = new StringBuffer(); | 
|  | writeDeclarationOn(method, buffer); | 
|  | if (method.isAbstract) buffer.write(', abstract'); | 
|  | if (method.isSynthetic) buffer.write(', synthetic'); | 
|  | if (method.isStatic) buffer.write(', static'); | 
|  | if (method.isGetter) buffer.write(', getter'); | 
|  | if (method.isSetter) buffer.write(', setter'); | 
|  | if (method.isConstructor) buffer.write(', constructor'); | 
|  | return 'Method($buffer)'; | 
|  | } | 
|  |  | 
|  | String stringifyDependencies(LibraryMirror l) { | 
|  | n(s) => s is Symbol ? MirrorSystem.getName(s) : s; | 
|  | int compareDep(a, b) { | 
|  | if (a.targetLibrary == b.targetLibrary) { | 
|  | if ((a.prefix != null) && (b.prefix != null)) { | 
|  | return n(a.prefix).compareTo(n(b.prefix)); | 
|  | } | 
|  | return a.prefix == null ? 1 : -1; | 
|  | } | 
|  | return n(a.targetLibrary.simpleName) | 
|  | .compareTo(n(b.targetLibrary.simpleName)); | 
|  | } | 
|  |  | 
|  | int compareCom(a, b) => n(a.identifier).compareTo(n(b.identifier)); | 
|  | int compareFirst(a, b) => a[0].compareTo(b[0]); | 
|  | sortBy(c, p) => new List.from(c)..sort(p); | 
|  |  | 
|  | var buffer = new StringBuffer(); | 
|  | sortBy(l.libraryDependencies, compareDep).forEach((dep) { | 
|  | if (dep.isImport) buffer.write('import '); | 
|  | if (dep.isExport) buffer.write('export '); | 
|  | buffer.write(n(dep.targetLibrary.simpleName)); | 
|  | if (dep.isDeferred) buffer.write(' deferred'); | 
|  | if (dep.prefix != null) buffer.write(' as ${n(dep.prefix)}'); | 
|  | buffer.write('\n'); | 
|  |  | 
|  | var flattenedCombinators = <dynamic>[]; | 
|  | dep.combinators.forEach((com) { | 
|  | com.identifiers.forEach((ident) { | 
|  | flattenedCombinators.add([n(ident), com.isShow, com.isHide]); | 
|  | }); | 
|  | }); | 
|  | sortBy(flattenedCombinators, compareFirst).forEach((triple) { | 
|  | buffer.write(' '); | 
|  | if (triple[1]) buffer.write('show '); | 
|  | if (triple[2]) buffer.write('hide '); | 
|  | buffer.write(triple[0]); | 
|  | buffer.write('\n'); | 
|  | }); | 
|  | }); | 
|  | return buffer.toString(); | 
|  | } | 
|  |  | 
|  | String stringify(value) { | 
|  | if (value == null) return '<null>'; | 
|  | if (value is Map) return stringifyMap(value); | 
|  | if (value is Iterable) return stringifyIterable(value); | 
|  | if (value is InstanceMirror) return stringifyInstance(value); | 
|  | if (value is ParameterMirror) return stringifyParameter(value); | 
|  | if (value is VariableMirror) return stringifyVariable(value); | 
|  | if (value is MethodMirror) return stringifyMethod(value); | 
|  | if (value is num) return value.toString(); | 
|  | if (value is String) return value; | 
|  | if (value is Symbol) return stringifySymbol(value); | 
|  | if (value is ClassMirror) return stringifyClass(value); | 
|  | if (value is TypeVariableMirror) return stringifyTypeVariable(value); | 
|  | if (value is TypeMirror) return stringifyType(value); | 
|  | throw 'Unexpected value: $value'; | 
|  | } | 
|  |  | 
|  | void expect(expected, actual, [String reason = ""]) { | 
|  | Expect.stringEquals(expected, stringify(actual), reason); | 
|  | } | 
|  |  | 
|  | int compareSymbols(Symbol a, Symbol b) { | 
|  | return MirrorSystem.getName(a).compareTo(MirrorSystem.getName(b)); | 
|  | } | 
|  |  | 
|  | Iterable<Symbol> simpleNames(Iterable<Mirror> i) => | 
|  | i.map((e) => (e as DeclarationMirror).simpleName); | 
|  |  | 
|  | List<Symbol> sort(Iterable<Symbol> symbols) => | 
|  | symbols.toList()..sort(compareSymbols); |