Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(94)

Side by Side Diff: pkg/metatest/lib/metatest.dart

Issue 524153002: Sharing metatest logic between unittest and scheduled_test (Closed) Base URL: https://dart.googlecode.com/svn/branches/bleeding_edge/dart
Patch Set: final tweaks Created 6 years, 3 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
OLDNEW
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file 1 // Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file
2 // for details. All rights reserved. Use of this source code is governed by a 2 // for details. All rights reserved. Use of this source code is governed by a
3 // BSD-style license that can be found in the LICENSE file. 3 // BSD-style license that can be found in the LICENSE file.
4 4
5 /// A test library for testing test libraries? We must go deeper. 5 /// A test library for testing test libraries? We must go deeper.
6 /// 6 ///
7 /// Since unit testing code tends to use a lot of global state, it can be tough 7 /// Since unit testing code tends to use a lot of global state, it can be tough
8 /// to test. This library manages it by running each test case in a child 8 /// to test. This library manages it by running each test case in a child
9 /// isolate, then reporting the results back to the parent isolate. 9 /// isolate, then reporting the results back to the parent isolate.
10 library metatest; 10 library metatest;
11 11
12 import 'dart:async'; 12 import 'dart:async';
13 import 'dart:io';
14 import 'dart:isolate'; 13 import 'dart:isolate';
15 14
16 import 'package:path/path.dart' as path;
17 import 'package:unittest/unittest.dart'; 15 import 'package:unittest/unittest.dart';
18 import 'package:scheduled_test/scheduled_test.dart' as scheduled_test;
19 16
20 import 'utils.dart'; 17 import 'src/utils.dart';
18
19 /// Whether or not we're running in a child isolate that's supposed to run a
20 /// test.
21 bool _inChildIsolate;
22
23 /// The port with which the child isolate should communicate with the parent
24 /// isolate.
25 ///
26 /// `null` in the parent isolate.
27 SendPort _replyTo;
28
29 /// The only value of the configuration used in metatest.
30 final _metaConfiguration = new _MetaConfiguration();
31
32 /// The function holding the tests to be run.
33 Function _testBody;
34
35 /// The description of the test to run in the child isolate.
36 ///
37 /// `null` in the parent isolate.
38 String _testToRun;
39
40 /// Stores the optional timeout used to override the default unittest timeout.
41 Duration _timeoutOverride;
42
43 /// Runs [setUpFn] before every metatest.
44 ///
45 /// Note that [setUpFn] will be overwritten if the test itself calls [setUp].
46 void metaSetUp(void setUpFn()) {
47 if (_inChildIsolate) setUp(setUpFn);
48 }
49
50 /// Runs a set of tests defined in `body` and checks the result by comparing
51 /// with values in `expectedResults`.
52 ///
53 /// [expectedResults] is a list which should have a [Map] value for each test
54 /// that is run. Each [Map] key corresponds to values from a completed test
55 /// case: "description", "message", "result", and "stackTrace".
nweiz 2014/09/17 22:39:16 Explain what the possible formats of "result" are.
kevmoo 2014/09/18 22:11:33 Done, although I was not specific about where the
56 ///
57 /// Here's an example of a `expectedResults` value for two tests, where the
58 /// where the first fails and the second passes.
59 ///
60 /// ```dart
61 /// [{
62 /// 'description': 'test',
63 /// 'message': 'Caught error!',
64 /// 'result': 'fail',
65 /// }, {
66 /// 'description': 'follow up',
67 /// 'result': 'pass',
68 /// }]
69 /// ```
70 void expectTestResults(String description, void body(),
71 List<Map> expectedResults) {
72 _setUpTest(description, body, (resultsMap) {
73 var list = resultsMap['results'];
74 expect(list, hasLength(expectedResults.length),
75 reason: 'The number of tests run does not match the numbef of expected'
nweiz 2014/09/17 22:39:16 "numbef" -> "number"
kevmoo 2014/09/18 22:11:33 Done.
76 ' results.');
77
78 for (var i = 0; i < list.length; i++) {
79 var expectedMap = expectedResults[i];
80 var map = list[i];
81
82 expectedMap.forEach((key, value) {
83 expect(map, containsPair(key, value), reason: 'A test did not match the'
84 ' expected value for "$key" at index $i.');
85 });
86 }
87 });
88 }
21 89
22 /// Declares a test with the given [description] and [body]. [body] corresponds 90 /// Declares a test with the given [description] and [body]. [body] corresponds
23 /// to the `main` method of a test file, and will be run in an isolate. By 91 /// to the `main` method of a test file, and will be run in an isolate. By
24 /// default, this expects that all tests defined in [body] pass, but if 92 /// default, this expects that all tests defined in [body] pass, but if
25 /// [passing] is passed, only tests listed there are expected to pass. 93 /// [passing] is passed, only tests listed there are expected to pass.
26 void expectTestsPass(String description, void body(), {List<String> passing}) { 94 void expectTestsPass(String description, void body(), {List<String> passing}) {
27 _setUpTest(description, body, (results) { 95 _setUpTest(description, body, (results) {
28 if (_hasError(results)) { 96 if (_hasError(results)) {
29 throw 'Expected all tests to pass, but got error(s):\n' 97 fail('Expected all tests to pass, but got error(s):\n'
30 '${_summarizeTests(results)}'; 98 '${_summarizeTests(results)}');
31 } else if (passing == null) { 99 } else if (passing == null) {
32 if (results['failed'] != 0) { 100 if (results['failed'] != 0) {
33 throw 'Expected all tests to pass, but some failed:\n' 101 fail('Expected all tests to pass, but some failed:\n'
34 '${_summarizeTests(results)}'; 102 '${_summarizeTests(results)}');
35 } 103 }
36 } else { 104 } else {
37 var shouldPass = new Set.from(passing); 105 var shouldPass = new Set.from(passing);
38 var didPass = new Set.from(results['results'] 106 var didPass = new Set.from(results['results']
39 .where((t) => t['result'] == 'pass') 107 .where((t) => t['result'] == 'pass')
40 .map((t) => t['description'])); 108 .map((t) => t['description']));
41 109
42 if (!shouldPass.containsAll(didPass) || 110 if (!shouldPass.containsAll(didPass) ||
43 !didPass.containsAll(shouldPass)) { 111 !didPass.containsAll(shouldPass)) {
44 String stringify(Set<String> tests) => 112 String stringify(Set<String> tests) =>
45 '{${tests.map((t) => '"$t"').join(', ')}}'; 113 '{${tests.map((t) => '"$t"').join(', ')}}';
46 114
47 fail('Expected exactly ${stringify(shouldPass)} to pass, but ' 115 fail('Expected exactly ${stringify(shouldPass)} to pass, but '
48 '${stringify(didPass)} passed.\n' 116 '${stringify(didPass)} passed.\n'
49 '${_summarizeTests(results)}'); 117 '${_summarizeTests(results)}');
50 } 118 }
51 } 119 }
52 }); 120 });
53 } 121 }
54 122
55 /// Declares a test with the given [description] and [body].
56 ///
57 /// [body] corresponds
58 /// to the `main` method of a test file, and will be run in an isolate.
59 ///
60 /// All tests must have an expected result in [expectedResults].
61 void expectTests(String description, void body(),
62 Map<String, String> expectedResults) {
63 _setUpTest(description, body, (results) {
64 expectedResults = new Map.from(expectedResults);
65
66 for (var testResult in results['results']) {
67 var description = testResult['description'];
68
69 expect(expectedResults, contains(description),
70 reason: '"$description" did not have an expected result set.\n'
71 '${_summarizeTests(results)}');
72
73 var result = testResult['result'];
74
75 expect(expectedResults, containsPair(description, result),
76 reason: 'The test "$description" not not have the expected result.\n'
77 '${_summarizeTests(results)}');
78
79 expectedResults.remove(description);
80 }
81 expect(expectedResults, isEmpty,
82 reason: 'Unexpected additional test results\n'
83 '${_summarizeTests(results)}');
84 });
85 }
86
87 /// Declares a test with the given [description] and [body]. [body] corresponds 123 /// Declares a test with the given [description] and [body]. [body] corresponds
88 /// to the `main` method of a test file, and will be run in an isolate. Expects 124 /// to the `main` method of a test file, and will be run in an isolate. Expects
89 /// all tests defined by [body] to fail. 125 /// all tests defined by [body] to fail.
90 void expectTestsFail(String description, void body()) { 126 void expectTestsFail(String description, void body()) {
91 _setUpTest(description, body, (results) { 127 _setUpTest(description, body, (results) {
92 if (_hasError(results)) { 128 if (_hasError(results)) {
93 throw 'Expected all tests to fail, but got error(s):\n' 129 throw 'Expected all tests to fail, but got error(s):\n'
94 '${_summarizeTests(results)}'; 130 '${_summarizeTests(results)}';
95 } else if (results['passed'] != 0) { 131 } else if (results['passed'] != 0) {
96 throw 'Expected all tests to fail, but some passed:\n' 132 throw 'Expected all tests to fail, but some passed:\n'
97 '${_summarizeTests(results)}'; 133 '${_summarizeTests(results)}';
98 } 134 }
99 }); 135 });
100 } 136 }
101 137
102 /// Runs [setUpFn] before every metatest. Note that [setUpFn] will be
103 /// overwritten if the test itself calls [setUp].
104 void metaSetUp(void setUpFn()) {
105 if (_inChildIsolate) scheduled_test.setUp(setUpFn);
106 }
107
108 /// Sets up a test with the given [description] and [body]. After the test runs, 138 /// Sets up a test with the given [description] and [body]. After the test runs,
109 /// calls [validate] with the result map. 139 /// calls [validate] with the result map.
110 void _setUpTest(String description, void body(), void validate(Map)) { 140 void _setUpTest(String description, void body(), void validate(Map map)) {
111 if (_inChildIsolate) { 141 if (_inChildIsolate) {
112 _ensureInitialized(); 142 _ensureInitialized();
113 if (_testToRun == description) body(); 143 if (_testToRun == description) body();
114 } else { 144 } else {
115 test(description, () { 145 test(description, () {
116 expect(_runInIsolate(description).then(validate), completes); 146 return _runInIsolate(description).then(validate);
117 }); 147 });
118 } 148 }
119 } 149 }
120 150
121 /// The description of the test to run in the child isolate.
122 ///
123 /// `null` in the parent isolate.
124 String _testToRun;
125
126 /// The port with which the child isolate should communicate with the parent
127 /// isolate.
128 ///
129 /// `null` in the parent isolate.
130 SendPort _replyTo;
131
132 /// Whether or not we're running in a child isolate that's supposed to run a
133 /// test.
134 bool _inChildIsolate;
135
136 /// Initialize metatest. 151 /// Initialize metatest.
137 /// 152 ///
138 /// [message] should be the second argument to [main]. It's used to determine 153 /// [message] should be the second argument to [main]. It's used to determine
139 /// whether this test is in the parent isolate or a child isolate. 154 /// whether this test is in the parent isolate or a child isolate.
140 void initMetatest(message) { 155 ///
156 /// [timeout], when specified, overrides the default timeout for unittest.
157 void initMetatest(message, {Duration timeout}) {
158 _timeoutOverride = timeout;
141 if (message == null) { 159 if (message == null) {
142 _inChildIsolate = false; 160 _inChildIsolate = false;
143 } else { 161 } else {
144 _testToRun = message['testToRun']; 162 _testToRun = message['testToRun'];
145 _replyTo = message['replyTo']; 163 _replyTo = message['replyTo'];
146 _inChildIsolate = true; 164 _inChildIsolate = true;
147 } 165 }
148 } 166 }
149 167
150 /// Runs the test described by [description] in its own isolate. Returns a map 168 // TODO(kevmoo) We need to capture the main method to allow running in an
151 /// describing the results of that test run. 169 // isolate. There is no mechanism to capture the current executing URI between
170 // browser and vm. Issue 1145 and/or Issue 8440
171 void initTests(void testBody(message)) {
172 _testBody = testBody;
173 _testBody(null);
174 }
175
176 /// Runs the test described by [description] in its own isolate.
177 ///
178 /// Returns a map describing the results of that test run.
152 Future<Map> _runInIsolate(String description) { 179 Future<Map> _runInIsolate(String description) {
180 if (_testBody == null) {
181 throw new StateError('initTests was not called.');
182 }
183
153 var replyPort = new ReceivePort(); 184 var replyPort = new ReceivePort();
154 // TODO(nweiz): Don't use path here once issue 8440 is fixed. 185 return Isolate.spawn(_testBody, {
155 var uri = path.toUri(path.absolute(path.fromUri(Platform.script)));
156 return Isolate.spawnUri(uri, [], {
157 'testToRun': description, 186 'testToRun': description,
158 'replyTo': replyPort.sendPort 187 'replyTo': replyPort.sendPort
159 }).then((_) { 188 }).then((_) => replyPort.first);
160 // TODO(nweiz): Remove this timeout once issue 8875 is fixed and we can
161 // capture top-level exceptions.
162 return timeout(replyPort.first, 30 * 1000, () {
163 throw 'Timed out waiting for test to complete.';
164 });
165 });
166 } 189 }
167 190
168 /// Returns whether [results] (a test result map) describes a test run in which 191 /// Returns whether [results] (a test result map) describes a test run in which
169 /// an error occurred. 192 /// an error occurred.
170 bool _hasError(Map results) { 193 bool _hasError(Map results) {
171 return results['errors'] > 0 || results['uncaughtError'] != null || 194 return results['errors'] > 0 || results['uncaughtError'] != null ||
172 (results['passed'] == 0 && results['failed'] == 0); 195 (results['passed'] == 0 && results['failed'] == 0);
173 } 196 }
174 197
175 /// Returns a string description of the test run descibed by [results]. 198 /// Returns a string description of the test run descibed by [results].
(...skipping 28 matching lines...) Expand all
204 return prefixLines(buffer.toString()); 227 return prefixLines(buffer.toString());
205 } 228 }
206 229
207 /// Indents each line of [str] by two spaces. 230 /// Indents each line of [str] by two spaces.
208 String _indent(String str) { 231 String _indent(String str) {
209 return str.replaceAll(new RegExp("^", multiLine: true), " "); 232 return str.replaceAll(new RegExp("^", multiLine: true), " ");
210 } 233 }
211 234
212 /// Ensure that the metatest configuration is loaded. 235 /// Ensure that the metatest configuration is loaded.
213 void _ensureInitialized() { 236 void _ensureInitialized() {
214 unittestConfiguration = _singleton; 237 unittestConfiguration = _metaConfiguration;
238 if (_timeoutOverride != null) {
239 unittestConfiguration.timeout = _timeoutOverride;
240 }
215 } 241 }
216 242
217 final _singleton = new _MetaConfiguration();
218
219 /// Special test configuration for use within the child isolates. This hides all 243 /// Special test configuration for use within the child isolates. This hides all
220 /// output and reports data back to the parent isolate. 244 /// output and reports data back to the parent isolate.
221 class _MetaConfiguration extends Configuration { 245 class _MetaConfiguration extends Configuration {
222 246
223 _MetaConfiguration() : super.blank(); 247 _MetaConfiguration() : super.blank();
224 248
225 void onSummary(int passed, int failed, int errors, List<TestCase> results, 249 void onSummary(int passed, int failed, int errors, List<TestCase> results,
226 String uncaughtError) { 250 String uncaughtError) {
227 _replyTo.send({ 251 _replyTo.send({
228 "passed": passed, 252 "passed": passed,
229 "failed": failed, 253 "failed": failed,
230 "errors": errors, 254 "errors": errors,
231 "uncaughtError": uncaughtError, 255 "uncaughtError": uncaughtError,
232 "results": results.map((testCase) => { 256 "results": results.map((testCase) => {
233 "description": testCase.description, 257 "description": testCase.description,
234 "message": testCase.message, 258 "message": testCase.message,
235 "result": testCase.result, 259 "result": testCase.result,
236 "stackTrace": testCase.stackTrace.toString() 260 "stackTrace": testCase.stackTrace.toString()
237 }).toList() 261 }).toList()
238 }); 262 });
239 } 263 }
240 } 264 }
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698