test/tools/javac/processing/environment/round/TestElementsAnnotatedWith.java

Mon, 31 Aug 2009 19:43:06 -0700

author
jjg
date
Mon, 31 Aug 2009 19:43:06 -0700
changeset 395
5a72ba18c471
parent 232
1fbc1cc6e260
child 453
96c71cbc544b
permissions
-rw-r--r--

6877759: test/tools/javac/processing/environment/round/TestElementsAnnotatedWith.java leaves open file
Reviewed-by: darcy

     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  */
    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  */
    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.*;
    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 {
    64     public boolean process(Set<? extends TypeElement> annotations,
    65                            RoundEnvironment roundEnvironment) {
    66         Elements elementUtils = processingEnv.getElementUtils();
    68         TypeElement annotatedElementInfoElement =
    69             elementUtils.getTypeElement("AnnotatedElementInfo");
    70         Set<? extends Element> resultsMeta = Collections.emptySet();
    71         Set<? extends Element> resultsBase = Collections.emptySet();
    73         if (!roundEnvironment.processingOver()) {
    74             testNonAnnotations(roundEnvironment);
    76             // Verify AnnotatedElementInfo is present on the first
    77             // specified type.
    79             TypeElement firstType = typesIn(roundEnvironment.getRootElements()).iterator().next();
    81             AnnotatedElementInfo annotatedElementInfo = firstType.getAnnotation(AnnotatedElementInfo.class);
    83             boolean failed = false;
    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.
    92                 Set<String> expectedNames = new HashSet<String>(Arrays.asList(annotatedElementInfo.names()));
    94                 resultsMeta =
    95                     roundEnvironment.
    96                     getElementsAnnotatedWith(elementUtils.
    97                                              getTypeElement(annotatedElementInfo.
    98                                                             annotationName())) ;
   100                 System.err.println("Results: " + resultsMeta);
   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             }
   117             resultsBase = computeResultsBase(roundEnvironment, annotatedElementInfo.annotationName());
   119             if (!resultsMeta.equals(resultsBase)) {
   120                 failed = true;
   121                 System.err.println("Base and Meta sets unequal;\n meta: " + resultsMeta +
   122                                    "\nbase: " + resultsBase);
   123             }
   125             if (failed) {
   126                 System.err.println("AnnotatedElementInfo: " + annotatedElementInfo);
   127                 throw new RuntimeException();
   128             }
   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);
   142         }
   143         return true;
   144     }
   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     }
   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) {}
   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     }
   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             try {
   192                 int datum = io.read();
   193                 while(datum != -1) {
   194                     os.write(datum);
   195                     datum = io.read();
   196                 }
   197             } finally {
   198                 io.close();
   199             }
   200             os.close();
   201         } catch (IOException ioe) {
   202             throw new RuntimeException(ioe);
   203         }
   206     }
   208     @Override
   209     public SourceVersion getSupportedSourceVersion() {
   210         return SourceVersion.latest();
   211     }
   212 }

mercurial