1 /* 2 * Copyright (C) 2018 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.tradefed.util; 18 19 import com.android.tradefed.result.MetricsXMLResultReporter; 20 import com.android.tradefed.result.TestDescription; 21 import com.android.tradefed.testtype.metricregression.Metrics; 22 23 import com.google.common.annotations.VisibleForTesting; 24 25 import org.xml.sax.Attributes; 26 import org.xml.sax.SAXException; 27 import org.xml.sax.helpers.DefaultHandler; 28 29 import java.io.BufferedInputStream; 30 import java.io.File; 31 import java.io.FileInputStream; 32 import java.io.IOException; 33 import java.io.InputStream; 34 import java.util.List; 35 import java.util.Set; 36 37 import javax.xml.parsers.ParserConfigurationException; 38 import javax.xml.parsers.SAXParser; 39 import javax.xml.parsers.SAXParserFactory; 40 41 /** Parser that extracts test metrics result data generated by {@link MetricsXMLResultReporter}. */ 42 public class MetricsXmlParser { 43 44 /** Thrown when MetricsXmlParser fails to parse a metrics xml file. */ 45 public static class ParseException extends Exception { ParseException(Throwable cause)46 public ParseException(Throwable cause) { 47 super(cause); 48 } 49 ParseException(String msg, Throwable cause)50 public ParseException(String msg, Throwable cause) { 51 super(msg, cause); 52 } 53 } 54 55 /* 56 * Parses the xml format. Expected tags/attributes are: 57 * testsuite name="runname" tests="X" 58 * runmetric name="metric1" value="1.0" 59 * testcase classname="FooTest" testname="testMethodName" 60 * testmetric name="metric2" value="1.0" 61 */ 62 private static class MetricsXmlHandler extends DefaultHandler { 63 64 private static final String TESTSUITE_TAG = "testsuite"; 65 private static final String TESTCASE_TAG = "testcase"; 66 private static final String TIME_TAG = "time"; 67 private static final String RUNMETRIC_TAG = "runmetric"; 68 private static final String TESTMETRIC_TAG = "testmetric"; 69 70 private TestDescription mCurrentTest = null; 71 72 private Metrics mMetrics; 73 private Set<String> mBlacklistMetrics; 74 MetricsXmlHandler(Metrics metrics, Set<String> blacklistMetrics)75 public MetricsXmlHandler(Metrics metrics, Set<String> blacklistMetrics) { 76 mMetrics = metrics; 77 mBlacklistMetrics = blacklistMetrics; 78 } 79 80 @Override startElement(String uri, String localName, String name, Attributes attributes)81 public void startElement(String uri, String localName, String name, Attributes attributes) 82 throws SAXException { 83 if (TESTSUITE_TAG.equalsIgnoreCase(name)) { 84 // top level tag - maps to a test run in TF terminology 85 String testCount = getMandatoryAttribute(name, "tests", attributes); 86 mMetrics.setNumTests(Integer.parseInt(testCount)); 87 mMetrics.addRunMetric(TIME_TAG, getMandatoryAttribute(name, TIME_TAG, attributes)); 88 } 89 if (TESTCASE_TAG.equalsIgnoreCase(name)) { 90 // start of description of an individual test method 91 String testClassName = getMandatoryAttribute(name, "classname", attributes); 92 String methodName = getMandatoryAttribute(name, "testname", attributes); 93 mCurrentTest = new TestDescription(testClassName, methodName); 94 } 95 if (RUNMETRIC_TAG.equalsIgnoreCase(name)) { 96 String metricName = getMandatoryAttribute(name, "name", attributes); 97 String metricValue = getMandatoryAttribute(name, "value", attributes); 98 if (!mBlacklistMetrics.contains(metricName)) { 99 mMetrics.addRunMetric(metricName, metricValue); 100 } 101 } 102 if (TESTMETRIC_TAG.equalsIgnoreCase(name)) { 103 String metricName = getMandatoryAttribute(name, "name", attributes); 104 String metricValue = getMandatoryAttribute(name, "value", attributes); 105 if (!mBlacklistMetrics.contains(metricName)) { 106 mMetrics.addTestMetric(mCurrentTest, metricName, metricValue); 107 } 108 } 109 } 110 getMandatoryAttribute(String tagName, String attrName, Attributes attributes)111 private String getMandatoryAttribute(String tagName, String attrName, Attributes attributes) 112 throws SAXException { 113 String value = attributes.getValue(attrName); 114 if (value == null) { 115 throw new SAXException( 116 String.format( 117 "Malformed XML, could not find '%s' attribute in '%s'", 118 attrName, tagName)); 119 } 120 return value; 121 } 122 } 123 124 /** 125 * Parses xml data contained in given input files. 126 * 127 * @param blacklistMetrics ignore the metrics with these names 128 * @param strictMode whether to throw an exception when metric validation fails 129 * @param metricXmlFiles a list of metric xml files 130 * @return a Metric object containing metrics from all metric files 131 * @throws ParseException if input could not be parsed 132 */ parse( Set<String> blacklistMetrics, boolean strictMode, List<File> metricXmlFiles)133 public static Metrics parse( 134 Set<String> blacklistMetrics, boolean strictMode, List<File> metricXmlFiles) 135 throws ParseException { 136 Metrics metrics = new Metrics(strictMode); 137 for (File xml : metricXmlFiles) { 138 try (InputStream is = new BufferedInputStream(new FileInputStream(xml))) { 139 parse(metrics, blacklistMetrics, is); 140 } catch (Exception e) { 141 throw new ParseException("Unable to parse " + xml.getPath(), e); 142 } 143 } 144 metrics.validate(metricXmlFiles.size()); 145 return metrics; 146 } 147 148 @VisibleForTesting parse(Metrics metrics, Set<String> blacklistMetrics, InputStream is)149 public static Metrics parse(Metrics metrics, Set<String> blacklistMetrics, InputStream is) 150 throws ParseException { 151 try { 152 SAXParserFactory parserFactory = SAXParserFactory.newInstance(); 153 parserFactory.setNamespaceAware(true); 154 SAXParser parser = parserFactory.newSAXParser(); 155 parser.parse(is, new MetricsXmlHandler(metrics, blacklistMetrics)); 156 return metrics; 157 } catch (ParserConfigurationException | SAXException | IOException e) { 158 throw new ParseException(e); 159 } 160 } 161 } 162