1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Eclipse Public License, Version 1.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.eclipse.org/org/documents/epl-v10.php
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 import org.w3c.dom.Document;
18 import org.w3c.dom.Element;
19 import org.w3c.dom.NamedNodeMap;
20 import org.w3c.dom.Node;
21 import org.w3c.dom.NodeList;
22 import org.xml.sax.InputSource;
23 import org.xml.sax.SAXException;
24 
25 import java.io.BufferedReader;
26 import java.io.BufferedWriter;
27 import java.io.File;
28 import java.io.FileNotFoundException;
29 import java.io.FileReader;
30 import java.io.FileWriter;
31 import java.io.IOException;
32 import java.io.Reader;
33 import java.io.StringReader;
34 import java.util.ArrayList;
35 import java.util.Collections;
36 import java.util.HashMap;
37 import java.util.HashSet;
38 import java.util.List;
39 import java.util.Map;
40 import java.util.Map.Entry;
41 import java.util.Set;
42 
43 import javax.xml.parsers.DocumentBuilder;
44 import javax.xml.parsers.DocumentBuilderFactory;
45 import javax.xml.parsers.ParserConfigurationException;
46 
47 /**
48  * Gathers statistics about attribute usage in layout files. This is how the "topAttrs"
49  * attributes listed in ADT's extra-view-metadata.xml (which drives the common attributes
50  * listed in the top of the context menu) is determined by running this script on a body
51  * of sample layout code.
52  * <p>
53  * This program takes one or more directory paths, and then it searches all of them recursively
54  * for layout files that are not in folders containing the string "test", and computes and
55  * prints frequency statistics.
56  */
57 public class Analyzer {
58     /** Number of attributes to print for each view */
59     public static final int ATTRIBUTE_COUNT = 6;
60     /** Separate out any attributes that constitute less than N percent of the total */
61     public static final int THRESHOLD = 10; // percent
62 
63     private List<File> mDirectories;
64     private File mCurrentFile;
65     private boolean mListAdvanced;
66 
67     /** Map from view id to map from attribute to frequency count */
68     private Map<String, Map<String, Usage>> mFrequencies =
69             new HashMap<String, Map<String, Usage>>(100);
70 
71     private Map<String, Map<String, Usage>> mLayoutAttributeFrequencies =
72             new HashMap<String, Map<String, Usage>>(100);
73 
74     private Map<String, String> mTopAttributes = new HashMap<String, String>(100);
75     private Map<String, String> mTopLayoutAttributes = new HashMap<String, String>(100);
76 
77     private int mFileVisitCount;
78     private int mLayoutFileCount;
79     private File mXmlMetadataFile;
80 
81     private Analyzer(List<File> directories, File xmlMetadataFile, boolean listAdvanced) {
82         mDirectories = directories;
83         mXmlMetadataFile = xmlMetadataFile;
84         mListAdvanced = listAdvanced;
85     }
86 
87     public static void main(String[] args) {
88         if (args.length < 1) {
89             System.err.println("Usage: " + Analyzer.class.getSimpleName()
90                     + " <directory1> [directory2 [directory3 ...]]\n");
91             System.err.println("Recursively scans for layouts in the given directory and");
92             System.err.println("computes statistics about attribute frequencies.");
93             System.exit(-1);
94         }
95 
96         File metadataFile = null;
97         List<File> directories = new ArrayList<File>();
98         boolean listAdvanced = false;
99         for (int i = 0, n = args.length; i < n; i++) {
100             String arg = args[i];
101 
102             if (arg.equals("--list")) {
103                 // List ALL encountered attributes
104                 listAdvanced = true;
105                 continue;
106             }
107 
108             // The -metadata flag takes a pointer to an ADT extra-view-metadata.xml file
109             // and attempts to insert topAttrs attributes into it (and saves it as same
110             // file +.mod as an extension). This isn't listed on the usage flag because
111             // it's pretty brittle and requires some manual fixups to the file afterwards.
112             if (arg.equals("--metadata")) {
113                 i++;
114                 File file = new File(args[i]);
115                 if (!file.exists()) {
116                     System.err.println(file.getName() + " does not exist");
117                     System.exit(-5);
118                 }
119                 if (!file.isFile() || !file.getName().endsWith(".xml")) {
120                     System.err.println(file.getName() + " must be an XML file");
121                     System.exit(-4);
122                 }
123                 metadataFile = file;
124                 continue;
125             }
126             File directory = new File(arg);
127             if (!directory.exists()) {
128                 System.err.println(directory.getName() + " does not exist");
129                 System.exit(-2);
130             }
131 
132             if (!directory.isDirectory()) {
133                 System.err.println(directory.getName() + " is not a directory");
134                 System.exit(-3);
135             }
136 
137             directories.add(directory);
138         }
139 
140         new Analyzer(directories, metadataFile, listAdvanced).analyze();
141     }
142 
143     private void analyze() {
144         for (File directory : mDirectories) {
145             scanDirectory(directory);
146         }
147 
148         if (mListAdvanced) {
149             listAdvanced();
150         }
151 
152         printStatistics();
153 
154         if (mXmlMetadataFile != null) {
155             printMergedMetadata();
156         }
157     }
158 
159     private void scanDirectory(File directory) {
160         File[] files = directory.listFiles();
161         if (files == null) {
162             return;
163         }
164 
165         for (File file : files) {
166             mFileVisitCount++;
167             if (mFileVisitCount % 50000 == 0) {
168                 System.out.println("Analyzed " + mFileVisitCount + " files...");
169             }
170 
171             if (file.isFile()) {
172                 scanFile(file);
173             } else if (file.isDirectory()) {
174                 // Skip stuff related to tests
175                 if (file.getName().contains("test")) {
176                     continue;
177                 }
178 
179                 // Recurse over subdirectories
180                 scanDirectory(file);
181             }
182         }
183     }
184 
185     private void scanFile(File file) {
186         if (file.getName().endsWith(".xml")) {
187             File parent = file.getParentFile();
188             if (parent.getName().startsWith("layout")) {
189                 analyzeLayout(file);
190             }
191         }
192 
193     }
194 
195     private void analyzeLayout(File file) {
196         mCurrentFile = file;
197         mLayoutFileCount++;
198         Document document = null;
199         DocumentBuilderFactory factory = DocumentBuilderFactory.newInstance();
200         InputSource is = new InputSource(new StringReader(readFile(file)));
201         try {
202             factory.setNamespaceAware(true);
203             factory.setValidating(false);
204             DocumentBuilder builder = factory.newDocumentBuilder();
205             document = builder.parse(is);
206 
207             analyzeDocument(document);
208 
209         } catch (ParserConfigurationException e) {
210             // pass -- ignore files we can't parse
211         } catch (SAXException e) {
212             // pass -- ignore files we can't parse
213         } catch (IOException e) {
214             // pass -- ignore files we can't parse
215         }
216     }
217 
218 
219     private void analyzeDocument(Document document) {
220         analyzeElement(document.getDocumentElement());
221     }
222 
223     private void analyzeElement(Element element) {
224         if (element.getTagName().equals("item")) {
225             // Resource files shouldn't be in the layout/ folder but I came across
226             // some cases
227             System.out.println("Warning: found <item> tag in a layout file in "
228                     + mCurrentFile.getPath());
229             return;
230         }
231 
232         countAttributes(element);
233         countLayoutAttributes(element);
234 
235         // Recurse over children
236         NodeList childNodes = element.getChildNodes();
237         for (int i = 0, n = childNodes.getLength(); i < n; i++) {
238             Node child = childNodes.item(i);
239             if (child.getNodeType() == Node.ELEMENT_NODE) {
240                 analyzeElement((Element) child);
241             }
242         }
243     }
244 
245     private void countAttributes(Element element) {
246         String tag = element.getTagName();
247         Map<String, Usage> attributeMap = mFrequencies.get(tag);
248         if (attributeMap == null) {
249             attributeMap = new HashMap<String, Usage>(70);
250             mFrequencies.put(tag, attributeMap);
251         }
252 
253         NamedNodeMap attributes = element.getAttributes();
254         for (int i = 0, n = attributes.getLength(); i < n; i++) {
255             Node attribute = attributes.item(i);
256             String name = attribute.getNodeName();
257 
258             if (name.startsWith("android:layout_")) {
259                 // Skip layout attributes; they are a function of the parent layout that this
260                 // view is embedded within, not the view itself.
261                 // TODO: Consider whether we should incorporate this info or make statistics
262                 // about that as well?
263                 continue;
264             }
265 
266             if (name.equals("android:id")) {
267                 // Skip ids: they are (mostly) unrelated to the view type and the tool
268                 // already offers id editing prominently
269                 continue;
270             }
271 
272             if (name.startsWith("xmlns:")) {
273                 // Unrelated to frequency counts
274                 continue;
275             }
276 
277             Usage usage = attributeMap.get(name);
278             if (usage == null) {
279                 usage = new Usage(name);
280             } else {
281                 usage.incrementCount();
282             }
283             attributeMap.put(name, usage);
284         }
285     }
286 
287     private void countLayoutAttributes(Element element) {
288         String parentTag = element.getParentNode().getNodeName();
289         Map<String, Usage> attributeMap = mLayoutAttributeFrequencies.get(parentTag);
290         if (attributeMap == null) {
291             attributeMap = new HashMap<String, Usage>(70);
292             mLayoutAttributeFrequencies.put(parentTag, attributeMap);
293         }
294 
295         NamedNodeMap attributes = element.getAttributes();
296         for (int i = 0, n = attributes.getLength(); i < n; i++) {
297             Node attribute = attributes.item(i);
298             String name = attribute.getNodeName();
299 
300             if (!name.startsWith("android:layout_")) {
301                 continue;
302             }
303 
304             // Skip layout_width and layout_height; they are mandatory in all but GridLayout so not
305             // very interesting
306             if (name.equals("android:layout_width") || name.equals("android:layout_height")) {
307                 continue;
308             }
309 
310             Usage usage = attributeMap.get(name);
311             if (usage == null) {
312                 usage = new Usage(name);
313             } else {
314                 usage.incrementCount();
315             }
316             attributeMap.put(name, usage);
317         }
318     }
319 
320     // Copied from AdtUtils
321     private static String readFile(File file) {
322         try {
323             return readFile(new FileReader(file));
324         } catch (FileNotFoundException e) {
325             e.printStackTrace();
326         }
327 
328         return null;
329     }
330 
331     private static String readFile(Reader inputStream) {
332         BufferedReader reader = null;
333         try {
334             reader = new BufferedReader(inputStream);
335             StringBuilder sb = new StringBuilder(2000);
336             while (true) {
337                 int c = reader.read();
338                 if (c == -1) {
339                     return sb.toString();
340                 } else {
341                     sb.append((char)c);
342                 }
343             }
344         } catch (IOException e) {
345             // pass -- ignore files we can't read
346         } finally {
347             try {
348                 if (reader != null) {
349                     reader.close();
350                 }
351             } catch (IOException e) {
352                 e.printStackTrace();
353             }
354         }
355 
356         return null;
357     }
358 
359     private void printStatistics() {
360         System.out.println("Analyzed " + mLayoutFileCount
361                 + " layouts (in a directory trees containing " + mFileVisitCount + " files)");
362         System.out.println("Top " + ATTRIBUTE_COUNT
363                 + " for each view (excluding layout_ attributes) :");
364         System.out.println("\n");
365         System.out.println(" Rank    Count    Share  Attribute");
366         System.out.println("=========================================================");
367         List<String> views = new ArrayList<String>(mFrequencies.keySet());
368         Collections.sort(views);
369         for (String view : views) {
370             String top = processUageMap(view, mFrequencies.get(view));
371             if (top != null) {
372                 mTopAttributes.put(view,  top);
373             }
374         }
375 
376         System.out.println("\n\n\nTop " + ATTRIBUTE_COUNT + " layout attributes (excluding "
377                 + "mandatory layout_width and layout_height):");
378         System.out.println("\n");
379         System.out.println(" Rank    Count    Share  Attribute");
380         System.out.println("=========================================================");
381         views = new ArrayList<String>(mLayoutAttributeFrequencies.keySet());
382         Collections.sort(views);
383         for (String view : views) {
384             String top = processUageMap(view, mLayoutAttributeFrequencies.get(view));
385             if (top != null) {
386                 mTopLayoutAttributes.put(view,  top);
387             }
388         }
389     }
390 
391     private static String processUageMap(String view, Map<String, Usage> map) {
392         if (map == null) {
393             return null;
394         }
395 
396         if (view.indexOf('.') != -1 && !view.startsWith("android.")) {
397             // Skip custom views
398             return null;
399         }
400 
401         List<Usage> values = new ArrayList<Usage>(map.values());
402         if (values.size() == 0) {
403             return null;
404         }
405 
406         Collections.sort(values);
407         int totalCount = 0;
408         for (Usage usage : values) {
409             totalCount += usage.count;
410         }
411 
412         System.out.println("\n<" + view + ">:");
413         if (view.equals("#document")) {
414             System.out.println("(Set on root tag, probably intended for included context)");
415         }
416 
417         int place = 1;
418         int count = 0;
419         int prevCount = -1;
420         float prevPercentage = 0f;
421         StringBuilder sb = new StringBuilder();
422         for (Usage usage : values) {
423             if (count++ >= ATTRIBUTE_COUNT && usage.count < prevCount) {
424                 break;
425             }
426 
427             float percentage = 100 * usage.count/(float)totalCount;
428             if (percentage < THRESHOLD && prevPercentage >= THRESHOLD) {
429                 System.out.println("  -----Less than 10%-------------------------------------");
430             }
431             System.out.printf("  %1d.    %5d    %5.1f%%  %s\n", place, usage.count,
432                     percentage, usage.attribute);
433 
434             prevPercentage = percentage;
435             if (prevCount != usage.count) {
436                 prevCount = usage.count;
437                 place++;
438             }
439 
440             if (percentage >= THRESHOLD /*&& usage.count > 1*/) { // 1:Ignore when not enough data?
441                 if (sb.length() > 0) {
442                     sb.append(',');
443                 }
444                 String name = usage.attribute;
445                 if (name.startsWith("android:")) {
446                     name = name.substring("android:".length());
447                 }
448                 sb.append(name);
449             }
450         }
451 
452         return sb.length() > 0 ? sb.toString() : null;
453     }
454 
455     private void printMergedMetadata() {
456         assert mXmlMetadataFile != null;
457         String metadata = readFile(mXmlMetadataFile);
458         if (metadata == null || metadata.length() == 0) {
459             System.err.println("Invalid metadata file");
460             System.exit(-6);
461         }
462 
463         System.err.flush();
464         System.out.println("\n\nUpdating layout metadata file...");
465         System.out.flush();
466 
467         StringBuilder sb = new StringBuilder((int) (2 * mXmlMetadataFile.length()));
468         String[] lines = metadata.split("\n");
469         for (int i = 0; i < lines.length; i++) {
470             String line = lines[i];
471             sb.append(line).append('\n');
472             int classIndex = line.indexOf("class=\"");
473             if (classIndex != -1) {
474                 int start = classIndex + "class=\"".length();
475                 int end = line.indexOf('"', start + 1);
476                 if (end != -1) {
477                     String view = line.substring(start, end);
478                     if (view.startsWith("android.widget.")) {
479                         view = view.substring("android.widget.".length());
480                     } else if (view.startsWith("android.view.")) {
481                         view = view.substring("android.view.".length());
482                     } else if (view.startsWith("android.webkit.")) {
483                         view = view.substring("android.webkit.".length());
484                     }
485                     String top = mTopAttributes.get(view);
486                     if (top == null) {
487                         System.err.println("Warning: No frequency data for view " + view);
488                     } else {
489                         sb.append(line.substring(0, classIndex)); // Indentation
490 
491                         sb.append("topAttrs=\"");
492                         sb.append(top);
493                         sb.append("\"\n");
494                     }
495 
496                     top = mTopLayoutAttributes.get(view);
497                     if (top != null) {
498                         // It's a layout attribute
499                         sb.append(line.substring(0, classIndex)); // Indentation
500 
501                         sb.append("topLayoutAttrs=\"");
502                         sb.append(top);
503                         sb.append("\"\n");
504                     }
505                 }
506             }
507         }
508 
509         System.out.println("\nTop attributes:");
510         System.out.println("--------------------------");
511         List<String> views = new ArrayList<String>(mTopAttributes.keySet());
512         Collections.sort(views);
513         for (String view : views) {
514             String top = mTopAttributes.get(view);
515             System.out.println(view + ": " + top);
516         }
517 
518         System.out.println("\nTop layout attributes:");
519         System.out.println("--------------------------");
520         views = new ArrayList<String>(mTopLayoutAttributes.keySet());
521         Collections.sort(views);
522         for (String view : views) {
523             String top = mTopLayoutAttributes.get(view);
524             System.out.println(view + ": " + top);
525         }
526 
527         System.out.println("\nModified XML metadata file:\n");
528         String newContent = sb.toString();
529         File output = new File(mXmlMetadataFile.getParentFile(), mXmlMetadataFile.getName() + ".mod");
530         if (output.exists()) {
531             output.delete();
532         }
533         try {
534             BufferedWriter writer = new BufferedWriter(new FileWriter(output));
535             writer.write(newContent);
536             writer.close();
537         } catch (IOException e) {
538             e.printStackTrace();
539         }
540         System.out.println("Done - wrote " + output.getPath());
541     }
542 
543     //private File mPublicFile = new File(location, "data/res/values/public.xml");
544     private File mPublicFile = new File("/Volumes/AndroidWork/git/frameworks/base/core/res/res/values/public.xml");
545 
546     private void listAdvanced() {
547         Set<String> keys = new HashSet<String>(1000);
548 
549         // Merged usages across view types
550         Map<String, Usage> mergedUsages = new HashMap<String, Usage>(100);
551 
552         for (Entry<String,Map<String,Usage>> entry : mFrequencies.entrySet()) {
553             String view = entry.getKey();
554             if (view.indexOf('.') != -1 && !view.startsWith("android.")) {
555                 // Skip custom views etc
556                 continue;
557             }
558             Map<String, Usage> map = entry.getValue();
559             for (Usage usage : map.values()) {
560 //                if (usage.count == 1) {
561 //                    System.out.println("Only found *one* usage of " + usage.attribute);
562 //                }
563 //                if (usage.count < 4) {
564 //                    System.out.println("Only found " + usage.count + " usage of " + usage.attribute);
565 //                }
566 
567                 String attribute = usage.attribute;
568                 int index = attribute.indexOf(':');
569                 if (index == -1 || attribute.startsWith("android:")) {
570                     Usage merged = mergedUsages.get(attribute);
571                     if (merged == null) {
572                         merged = new Usage(attribute);
573                         merged.count = usage.count;
574                         mergedUsages.put(attribute, merged);
575                     } else {
576                         merged.count += usage.count;
577                     }
578                 }
579             }
580         }
581 
582         for (Usage usage : mergedUsages.values())  {
583             String attribute = usage.attribute;
584             if (usage.count < 4) {
585                 System.out.println("Only found " + usage.count + " usage of " + usage.attribute);
586                 continue;
587             }
588             int index = attribute.indexOf(':');
589             if (index != -1) {
590                 attribute = attribute.substring(index + 1); // +1: skip ':'
591             }
592             keys.add(attribute);
593         }
594 
595         List<String> sorted = new ArrayList<String>(keys);
596         Collections.sort(sorted);
597         System.out.println("\nEncountered Attributes");
598         System.out.println("-----------------------------");
599         for (String attribute : sorted) {
600             System.out.println(attribute);
601         }
602 
603         System.out.println();
604     }
605 
606     private static class Usage implements Comparable<Usage> {
607         public String attribute;
608         public int count;
609 
610 
611         public Usage(String attribute) {
612             super();
613             this.attribute = attribute;
614 
615             count = 1;
616         }
617 
618         public void incrementCount() {
619             count++;
620         }
621 
622         @Override
623         public int compareTo(Usage o) {
624             // Sort by decreasing frequency, then sort alphabetically
625             int frequencyDelta = o.count - count;
626             if (frequencyDelta != 0) {
627                 return frequencyDelta;
628             } else {
629                 return attribute.compareTo(o.attribute);
630             }
631         }
632 
633         @Override
634         public String toString() {
635             return attribute + ": " + count;
636         }
637 
638         @Override
639         public int hashCode() {
640             final int prime = 31;
641             int result = 1;
642             result = prime * result + ((attribute == null) ? 0 : attribute.hashCode());
643             return result;
644         }
645 
646         @Override
647         public boolean equals(Object obj) {
648             if (this == obj)
649                 return true;
650             if (obj == null)
651                 return false;
652             if (getClass() != obj.getClass())
653                 return false;
654             Usage other = (Usage) obj;
655             if (attribute == null) {
656                 if (other.attribute != null)
657                     return false;
658             } else if (!attribute.equals(other.attribute))
659                 return false;
660             return true;
661         }
662     }
663 }
664