Index: third_party/typ/typ/json_results.py |
diff --git a/third_party/typ/typ/json_results.py b/third_party/typ/typ/json_results.py |
new file mode 100644 |
index 0000000000000000000000000000000000000000..8e03f6febbdc485bc9ccf2785e5369a21280b3c7 |
--- /dev/null |
+++ b/third_party/typ/typ/json_results.py |
@@ -0,0 +1,186 @@ |
+# Copyright 2014 Google Inc. All rights reserved. |
+# |
+# Licensed under the Apache License, Version 2.0 (the "License"); |
+# you may not use this file except in compliance with the License. |
+# You may obtain a copy of the License at |
+# |
+# http://www.apache.org/licenses/LICENSE-2.0 |
+# |
+# Unless required by applicable law or agreed to in writing, software |
+# distributed under the License is distributed on an "AS IS" BASIS, |
+# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
+# See the License for the specific language governing permissions and |
+# limitations under the License. |
+ |
+from collections import OrderedDict |
+ |
+import json |
+ |
+ |
+class ResultType(object): |
+ Pass = 'Pass' |
+ Failure = 'Failure' |
+ ImageOnlyFailure = 'ImageOnlyFailure' |
+ Timeout = 'Timeout' |
+ Crash = 'Crash' |
+ Skip = 'Skip' |
+ |
+ values = (Pass, Failure, ImageOnlyFailure, Timeout, Crash, Skip) |
+ |
+ |
+class Result(object): |
+ # too many instance attributes pylint: disable=R0902 |
+ # too many arguments pylint: disable=R0913 |
+ |
+ def __init__(self, name, actual, started, took, worker, |
+ expected=None, unexpected=False, |
+ flaky=False, code=0, out='', err='', pid=0): |
+ self.name = name |
+ self.actual = actual |
+ self.started = started |
+ self.took = took |
+ self.worker = worker |
+ self.expected = expected or [ResultType.Pass] |
+ self.unexpected = unexpected |
+ self.flaky = flaky |
+ self.code = code |
+ self.out = out |
+ self.err = err |
+ self.pid = pid |
+ |
+ |
+class ResultSet(object): |
+ |
+ def __init__(self): |
+ self.results = [] |
+ |
+ def add(self, result): |
+ self.results.append(result) |
+ |
+ |
+TEST_SEPARATOR = '.' |
+ |
+ |
+def make_full_results(metadata, seconds_since_epoch, all_test_names, results): |
+ """Convert the typ results to the Chromium JSON test result format. |
+ |
+ See http://www.chromium.org/developers/the-json-test-results-format |
+ """ |
+ |
+ # We use OrderedDicts here so that the output is stable. |
+ full_results = OrderedDict() |
+ full_results['version'] = 3 |
+ full_results['interrupted'] = False |
+ full_results['path_delimiter'] = TEST_SEPARATOR |
+ full_results['seconds_since_epoch'] = seconds_since_epoch |
+ |
+ for md in metadata: |
+ key, val = md.split('=', 1) |
+ full_results[key] = val |
+ |
+ passing_tests = _passing_test_names(results) |
+ failed_tests = failed_test_names(results) |
+ skipped_tests = set(all_test_names) - passing_tests - failed_tests |
+ |
+ full_results['num_failures_by_type'] = OrderedDict() |
+ full_results['num_failures_by_type']['FAIL'] = len(failed_tests) |
+ full_results['num_failures_by_type']['PASS'] = len(passing_tests) |
+ full_results['num_failures_by_type']['SKIP'] = len(skipped_tests) |
+ |
+ full_results['tests'] = OrderedDict() |
+ |
+ for test_name in all_test_names: |
+ value = OrderedDict() |
+ if test_name in skipped_tests: |
+ value['expected'] = 'SKIP' |
+ value['actual'] = 'SKIP' |
+ else: |
+ value['expected'] = 'PASS' |
+ value['actual'] = _actual_results_for_test(test_name, results) |
+ if value['actual'].endswith('FAIL'): |
+ value['is_unexpected'] = True |
+ _add_path_to_trie(full_results['tests'], test_name, value) |
+ |
+ return full_results |
+ |
+ |
+def make_upload_request(test_results_server, builder, master, testtype, |
+ full_results): |
+ url = 'http://%s/testfile/upload' % test_results_server |
+ attrs = [('builder', builder), |
+ ('master', master), |
+ ('testtype', testtype)] |
+ content_type, data = _encode_multipart_form_data(attrs, full_results) |
+ return url, content_type, data |
+ |
+ |
+def exit_code_from_full_results(full_results): |
+ return 1 if num_failures(full_results) else 0 |
+ |
+ |
+def num_failures(full_results): |
+ return full_results['num_failures_by_type']['FAIL'] |
+ |
+ |
+def failed_test_names(results): |
+ names = set() |
+ for r in results.results: |
+ if r.actual == ResultType.Failure: |
+ names.add(r.name) |
+ elif (r.actual == ResultType.Pass and |
+ r.name in names): # pragma: untested |
+ names.remove(r.name) |
+ return names |
+ |
+ |
+def _passing_test_names(results): |
+ return set(r.name for r in results.results if r.actual == ResultType.Pass) |
+ |
+ |
+def _actual_results_for_test(test_name, results): |
+ actuals = [] |
+ for r in results.results: |
+ if r.name == test_name: |
+ if r.actual == ResultType.Failure: |
+ actuals.append('FAIL') |
+ elif r.actual == ResultType.Pass: |
+ actuals.append('PASS') |
+ |
+ assert actuals, 'We did not find any result data for %s.' % test_name |
+ return ' '.join(actuals) |
+ |
+ |
+def _add_path_to_trie(trie, path, value): |
+ if TEST_SEPARATOR not in path: |
+ trie[path] = value |
+ return |
+ directory, rest = path.split(TEST_SEPARATOR, 1) |
+ if directory not in trie: |
+ trie[directory] = {} |
+ _add_path_to_trie(trie[directory], rest, value) |
+ |
+ |
+def _encode_multipart_form_data(attrs, test_results): |
+ # Cloned from webkitpy/common/net/file_uploader.py |
+ BOUNDARY = '-J-S-O-N-R-E-S-U-L-T-S---B-O-U-N-D-A-R-Y-' |
+ CRLF = '\r\n' |
+ lines = [] |
+ |
+ for key, value in attrs: |
+ lines.append('--' + BOUNDARY) |
+ lines.append('Content-Disposition: form-data; name="%s"' % key) |
+ lines.append('') |
+ lines.append(value) |
+ |
+ lines.append('--' + BOUNDARY) |
+ lines.append('Content-Disposition: form-data; name="file"; ' |
+ 'filename="full_results.json"') |
+ lines.append('Content-Type: application/json') |
+ lines.append('') |
+ lines.append(json.dumps(test_results)) |
+ |
+ lines.append('--' + BOUNDARY + '--') |
+ lines.append('') |
+ body = CRLF.join(lines) |
+ content_type = 'multipart/form-data; boundary=%s' % BOUNDARY |
+ return content_type, body |