1 /*
2  * Copyright 2010 Google Inc.
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License"); you may not
5  * use this file except in compliance with the License. You may obtain a copy of
6  * 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, WITHOUT
12  * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13  * License for the specific language governing permissions and limitations under
14  * the License.
15  */
16 
17 package com.android.accessibility;
18 
19 import org.xml.sax.InputSource;
20 import org.xml.sax.SAXException;
21 import org.xml.sax.XMLReader;
22 import org.xml.sax.helpers.XMLReaderFactory;
23 
24 import java.io.*;
25 import java.util.ArrayList;
26 import java.util.List;
27 import java.util.logging.Logger;
28 
29 /**
30  * An object that fetches all Android layout files and manages the testing of
31  * the files with the use of the AccessibilityValidationContentHandler. This
32  * object also reports on any errors encountered during the testing.
33  *
34  * @author dtseng@google.com (David Tseng)
35  */
36 public class AccessibilityValidator {
37     /** The root path to scan for Android layout files. */
38     private final File mRootFilePath;
39     /** Errors generated by thrown exceptions (and not by validation errors). */
40     private final List<String> mGeneralErrors = new ArrayList<String>();
41     /** A list of files we wish to have tested. */
42     private List<InputSource> mLayoutFiles;
43     /** The total number of validation test errors across all files. */
44     private int mTotalValidationErrors = 0;
45     /** The path to the Android sdk jar. */
46     private final File mAndroidSdkPath;
47 
48     /** The object that handles our logging. */
49     private static final Logger sLogger = Logger.getLogger("android.accessibility");
50 
51     /**
52      * The entry point to this tool.
53      *
54      * @param <args>
55      *            path on which to search for layout xml files that need
56      *            validation
57      */
58     public static void main(String[] args) {
59         sLogger.info("AccessibilityValidator");
60         if (args.length == 2) {
61             sLogger.info("Validating classes using android jar for subclasses of ImageView");
62             new AccessibilityValidator(args[0], args[1]).run();
63         } else {
64             sLogger.info("Usage: java AccessibilityValidator <path> <Android jar path>");
65             return;
66         }
67     }
68 
69     /**
70      * Constructs an AccessibilityValidator object using the root path and the
71      * android jar path.
72      */
73     public AccessibilityValidator(String rootPath, String androidSdkPath) {
74         mRootFilePath = new File(rootPath);
75         mAndroidSdkPath = new File(androidSdkPath);
76 
77         if (!mRootFilePath.exists()) {
78             throw new IllegalArgumentException("Invalid root path specified "
79                     + rootPath);
80         } else if (!mAndroidSdkPath.exists()) {
81             throw new IllegalArgumentException(
82                     "Invalid android sdk path specified " + androidSdkPath);
83         }
84     }
85 
86     /**
87      * Performs validation of Android layout files and logs errors that have
88      * been encountered during the validation. Returns true if the validation
89      * passes.
90      */
91     private boolean run() {
92         sLogger.info("Validating files under " + mRootFilePath);
93         mLayoutFiles = findLayoutFiles(mRootFilePath);
94         validateFiles();
95         for (String error : mGeneralErrors) {
96             sLogger.info(error);
97         }
98         sLogger.info("done with validation");
99         return mGeneralErrors.size() == 0;
100     }
101 
102     /**
103      * Accumulates a list of files under the path that meet two constraints.
104      * Firstly, it has a containing (parent) directory of "layout". Secondly, it
105      * has an xml extension
106      */
107     private List<InputSource> findLayoutFiles(File directory) {
108         List<InputSource> layoutFiles = new ArrayList<InputSource>();
109 
110         for (File file : directory.listFiles()) {
111             // The file is a directory; recurse on the file.
112             if (file.isDirectory()) {
113                 List<InputSource> directoryFiles = findLayoutFiles(file);
114                 layoutFiles.addAll(directoryFiles);
115                 // Does the containing directory and filename meet our
116                 // constraints?
117             } else if (directory.getName().toLowerCase().contains("layout")
118                     && file.getName().toLowerCase().endsWith(".xml")) {
119                 InputSource addition;
120                 try {
121                     addition = new InputSource(new FileReader(file));
122                     // Store this explicitly for logging.
123                     addition.setPublicId(file.toString());
124                     layoutFiles.add(addition);
125                 } catch (FileNotFoundException fileNotFoundException) {
126                     mGeneralErrors.add("File not found "
127                             + fileNotFoundException);
128                 }
129             }
130         }
131 
132         return layoutFiles;
133     }
134 
135     /*
136      * Processes a list of files via an AccessibilityValidationContentHandler.
137      * The caller will only be notified of errors via logging.
138      */
139     public void validateFiles() {
140         sLogger.info("Validating " + getLayoutFiles().size());
141         XMLReader reader;
142         try {
143             reader = XMLReaderFactory.createXMLReader();
144         } catch (SAXException saxExp) {
145             mGeneralErrors.add("Error " + saxExp);
146             return;
147         }
148         for (InputSource file : getLayoutFiles()) {
149             try {
150                 AccessibilityValidationContentHandler contentHandler
151                 = new AccessibilityValidationContentHandler(
152                         file.getPublicId(), mAndroidSdkPath);
153                 reader.setContentHandler(contentHandler);
154                 reader.parse(file);
155                 mTotalValidationErrors += contentHandler.getValidationErrors();
156             } catch (IOException ioExp) {
157                 mGeneralErrors.add("Error reading file " + ioExp);
158             } catch (SAXException saxExp) {
159                 mGeneralErrors.add("Error " + saxExp);
160             }
161         }
162     }
163 
164     /**
165      * Returns the number of general errors (considered caught exceptions).
166      */
167     public List<String> getGeneralErrors() {
168         return mGeneralErrors;
169     }
170 
171     /**
172      * Sets the files to be tested.
173      */
174     public void setLayoutFiles(List<InputSource> layoutFiles) {
175         this.mLayoutFiles = layoutFiles;
176     }
177 
178     /**
179      * Gets the files to be tested.
180      */
181     public List<InputSource> getLayoutFiles() {
182         return mLayoutFiles;
183     }
184 
185     /**
186      * Gets the total number of test validation errors across all files.
187      */
188     public int getTotalValidationErrors() {
189         return mTotalValidationErrors;
190     }
191 }
192