1 /* 2 * Copyright 2006-2009 Sun Microsystems, Inc. All Rights Reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. 8 * 9 * This code is distributed in the hope that it will be useful, but WITHOUT 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 12 * version 2 for more details (a copy is included in the LICENSE file that 13 * accompanied this code). 14 * 15 * You should have received a copy of the GNU General Public License version 16 * 2 along with this work; if not, write to the Free Software Foundation, 17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 18 * 19 * Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara, 20 * CA 95054 USA or visit www.sun.com if you need additional information or 21 * have any questions. 22 */ 23 24 /* 25 * @test 26 * @bug 6397298 6400986 6425592 6449798 6453386 6508401 6498938 27 * @summary Tests that getElementsAnnotatedWith works properly. 28 * @author Joseph D. Darcy 29 * @compile TestElementsAnnotatedWith.java 30 * @compile InheritedAnnotation.java 31 * @compile -processor TestElementsAnnotatedWith -proc:only SurfaceAnnotations.java 32 * @compile -processor TestElementsAnnotatedWith -proc:only BuriedAnnotations.java 33 * @compile -processor TestElementsAnnotatedWith -proc:only Part1.java Part2.java 34 * @compile -processor TestElementsAnnotatedWith -proc:only C2.java 35 * @compile -processor TestElementsAnnotatedWith -proc:only Foo.java 36 * @compile -XD-d=. Foo.java 37 * @compile -processor TestElementsAnnotatedWith -proc:only TestElementsAnnotatedWith.java 38 */ 39 40 import java.lang.annotation.Annotation; 41 import java.io.*; 42 import java.util.Collections; 43 import java.util.Set; 44 import java.util.HashSet; 45 import java.util.List; 46 import java.util.ArrayList; 47 import java.util.Arrays; 48 import javax.annotation.processing.*; 49 import javax.tools.*; 50 import javax.lang.model.SourceVersion; 51 import javax.lang.model.element.*; 52 import javax.lang.model.util.*; 53 import static javax.lang.model.util.ElementFilter.*; 54 55 /** 56 * This processor verifies that the information returned by 57 * getElementsAnnotatedWith is consistent with the expected results 58 * stored in an AnnotatedElementInfo annotation. 59 */ 60 @SupportedAnnotationTypes("*") 61 @AnnotatedElementInfo(annotationName="java.lang.SuppressWarnings", expectedSize=0, names={}) 62 public class TestElementsAnnotatedWith extends AbstractProcessor { 63 64 public boolean process(Set<? extends TypeElement> annotations, 65 RoundEnvironment roundEnvironment) { 66 Elements elementUtils = processingEnv.getElementUtils(); 67 68 TypeElement annotatedElementInfoElement = 69 elementUtils.getTypeElement("AnnotatedElementInfo"); 70 Set<? extends Element> resultsMeta = Collections.emptySet(); 71 Set<? extends Element> resultsBase = Collections.emptySet(); 72 73 if (!roundEnvironment.processingOver()) { 74 testNonAnnotations(roundEnvironment); 75 76 // Verify AnnotatedElementInfo is present on the first 77 // specified type. 78 79 TypeElement firstType = typesIn(roundEnvironment.getRootElements()).iterator().next(); 80 81 AnnotatedElementInfo annotatedElementInfo = firstType.getAnnotation(AnnotatedElementInfo.class); 82 83 boolean failed = false; 84 85 if (annotatedElementInfo == null) 86 throw new IllegalArgumentException("Missing AnnotatedElementInfo annotation on " + 87 firstType); 88 else { 89 // Verify that the annotation information is as 90 // expected. 91 92 Set<String> expectedNames = new HashSet<String>(Arrays.asList(annotatedElementInfo.names())); 93 94 resultsMeta = 95 roundEnvironment. 96 getElementsAnnotatedWith(elementUtils. 97 getTypeElement(annotatedElementInfo. 98 annotationName())) ; 99 100 System.err.println("Results: " + resultsMeta); 101 102 if (resultsMeta.size() != annotatedElementInfo.expectedSize()) { 103 failed = true; 104 System.err.printf("Bad number of elements; expected %d, got %d%n", 105 annotatedElementInfo.expectedSize(), resultsMeta.size()); 106 } else { 107 for(Element element : resultsMeta) { 108 String simpleName = element.getSimpleName().toString(); 109 if (!expectedNames.contains(simpleName) ) { 110 failed = true; 111 System.err.println("Name ``" + simpleName + "'' not expected."); 112 } 113 } 114 } 115 } 116 117 resultsBase = computeResultsBase(roundEnvironment, annotatedElementInfo.annotationName()); 118 119 if (!resultsMeta.equals(resultsBase)) { 120 failed = true; 121 System.err.println("Base and Meta sets unequal;\n meta: " + resultsMeta + 122 "\nbase: " + resultsBase); 123 } 124 125 if (failed) { 126 System.err.println("AnnotatedElementInfo: " + annotatedElementInfo); 127 throw new RuntimeException(); 128 } 129 130 if("TestElementsAnnotatedWith".equals(firstType.getSimpleName().toString())) 131 writeClassFile(); // Start another round to test class file input 132 } else { 133 // If processing is over without an error, the specified 134 // elements should be empty so an empty set should be returned. 135 resultsMeta = roundEnvironment.getElementsAnnotatedWith(annotatedElementInfoElement); 136 resultsBase = roundEnvironment.getElementsAnnotatedWith(AnnotatedElementInfo.class); 137 if (!resultsMeta.isEmpty()) 138 throw new RuntimeException("Nonempty resultsMeta: " + resultsMeta); 139 if (!resultsBase.isEmpty()) 140 throw new RuntimeException("Nonempty resultsBase: " + resultsBase); 141 142 } 143 return true; 144 } 145 146 private Set<? extends Element> computeResultsBase(RoundEnvironment roundEnvironment, String name) { 147 try { 148 return roundEnvironment. 149 getElementsAnnotatedWith(Class.forName(name).asSubclass(Annotation.class)); 150 } catch(ClassNotFoundException cnfe) { 151 throw new RuntimeException(cnfe); 152 } 153 } 154 155 /** 156 * Verify non-annotation types result in 157 * IllegalArgumentExceptions. 158 */ 159 private void testNonAnnotations(RoundEnvironment roundEnvironment) { 160 try { 161 Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith((Class)Object.class ); 162 throw new RuntimeException("Illegal argument exception not thrown"); 163 } catch(IllegalArgumentException iae) {} 164 165 try { 166 Set<? extends Element> elements = roundEnvironment.getElementsAnnotatedWith(processingEnv. 167 getElementUtils(). 168 getTypeElement("java.lang.Object") ); 169 throw new RuntimeException("Illegal argument exception not thrown"); 170 } catch(IllegalArgumentException iae) {} 171 } 172 173 /* 174 * Hack alert! The class file read below is generated by the 175 * "@compile -XD-d=. Foo.java" directive above. This sneakily 176 * overrides the output location to the current directory where a 177 * subsequent @compile can read the file. This could be improved 178 * if either a new directive like @process accepted class file 179 * arguments (the javac command accepts such arguments but 180 * @compile does not) or the test.src and test.classes properties 181 * were set to be read with @compile jobs. 182 */ 183 private void writeClassFile() { 184 try { 185 Filer filer = processingEnv.getFiler(); 186 JavaFileObject jfo = filer.createClassFile("Foo"); 187 OutputStream os = jfo.openOutputStream(); 188 // Copy the bytes over 189 System.out.println((new File(".")).getAbsolutePath()); 190 InputStream io = new BufferedInputStream(new FileInputStream(new File(".", "Foo.class"))); 191 int datum = io.read(); 192 while(datum != -1) { 193 os.write(datum); 194 datum = io.read(); 195 } 196 os.close(); 197 } catch (IOException io) { 198 throw new RuntimeException(io); 199 } 200 201 202 } 203 204 @Override 205 public SourceVersion getSupportedSourceVersion() { 206 return SourceVersion.latest(); 207 } 208 }