langtools/test/tools/javac/classfiles/attributes/innerclasses/InnerClassesInLocalClassTest.java
author jlahoda
Thu, 09 Oct 2014 10:08:52 +0200
changeset 26993 513b2cae81c3
parent 26103 3f96599b20da
child 30730 d3ce7619db2c
permissions -rw-r--r--
8057652: Request to improve error messages for labeled declarations Summary: Parse labeled statements as block statements to improve error recovery for labeled declarations; related cleanup. Reviewed-by: jjg

/*
 * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

/*
 * @test
 * @bug 8042251
 * @summary Testing InnerClasses_attribute of inner classes in local class.
 * @library /tools/lib /tools/javac/lib ../lib
 * @build InnerClassesTestBase TestResult TestBase InMemoryFileManager ToolBox
 * @run main InnerClassesInLocalClassTest
 */

import java.util.*;

public class InnerClassesInLocalClassTest extends InnerClassesTestBase {

    private final static Modifier[] LOCAL_CLASS_MODIFIERS =
            new Modifier[]{Modifier.EMPTY, Modifier.ABSTRACT, Modifier.FINAL};
    private final static String CLASS_TEMPLATE =
            "public %CLASS% OuterClass {\n" +
            "%SOURCE%\n" +
            "}";

    private final List<Data> innerClassesData;

    public InnerClassesInLocalClassTest() {
        innerClassesData = new ArrayList<>();
        for (Modifier outerModifier : LOCAL_CLASS_MODIFIERS) {
            StringBuilder sb = new StringBuilder();
            sb.append(outerModifier.getString()).append(' ');
            sb.append("class Local {");
            Map<String, Set<String>> class2Flags = new HashMap<>();
            for (int i = 0; i < LOCAL_CLASS_MODIFIERS.length; ++i) {
                Modifier innerModifier = LOCAL_CLASS_MODIFIERS[i];
                sb.append(innerModifier.getString()).append(' ')
                        .append("class").append(' ')
                        .append('A').append(i).append("{}\n");
                class2Flags.put("A" + i, getFlags(innerModifier));
            }
            sb.append("};");
            class2Flags.put("1Local", getFlags(outerModifier));
            innerClassesData.add(new Data(sb.toString(), class2Flags));
        }
    }

    public static void main(String[] args) throws TestFailedException {
        InnerClassesTestBase test = new InnerClassesInLocalClassTest();
        test.test("OuterClass$1Local", "1Local");
    }

    @Override
    public void setProperties() {
    }

    @Override
    public List<TestCase> generateTestCases() {
        List<TestCase> testCases = new ArrayList<>();
        testCases.addAll(localClassInClassMethod());
        testCases.addAll(localClassInInterfaceMethod());
        return testCases;
    }

    private List<TestCase> localClassInClassMethod() {
        List<TestCase> list = new ArrayList<>();
        String template = CLASS_TEMPLATE.replace("%CLASS%", "class");
        list.addAll(lambda(template));
        list.addAll(constructor(template));
        list.addAll(method(template,
                new Modifier[]{Modifier.EMPTY, Modifier.PRIVATE, Modifier.PROTECTED, Modifier.PUBLIC},
                new Modifier[]{Modifier.EMPTY, Modifier.FINAL, Modifier.STATIC}));
        list.addAll(staticAndInstanceInitializer(template));
        return list;
    }

    private List<TestCase> localClassInInterfaceMethod() {
        String template = CLASS_TEMPLATE.replace("%CLASS%", "interface");
        return method(template,
                new Modifier[]{Modifier.EMPTY, Modifier.PUBLIC},
                new Modifier[]{Modifier.DEFAULT, Modifier.STATIC});
    }

    private List<TestCase> generate(String template, String prefix, String suffix) {
        List<TestCase> list = new ArrayList<>();
        for (Data data : innerClassesData) {
            list.add(new TestCase(template.replace("%SOURCE%",
                    prefix + data.sources + suffix),
                    data.class2Flags));
        }
        return list;
    }

    private List<TestCase> lambda(String template) {
        return generate(template, "Runnable run = () -> {", "};");
    }

    private List<TestCase> constructor(String template) {
        List<TestCase> list = new ArrayList<>();
        for (Modifier modifier :
                new Modifier[]{Modifier.EMPTY, Modifier.PRIVATE, Modifier.PROTECTED, Modifier.PUBLIC}) {
            list.addAll(generate(template, modifier.getString() + " OuterClass() {", "}"));
        }
        return list;
    }

    private List<TestCase> method(String template, Modifier[] mods, Modifier[] otherMods) {
        List<TestCase> list = new ArrayList<>();
        for (Modifier modifier : mods) {
            for (Modifier otherMod : otherMods) {
                list.addAll(generate(template,
                        String.format("%s %s void method() {",
                                modifier.getString(),
                                otherMod.getString()),
                        "}"));
            }
        }
        return list;
    }

    private List<TestCase> staticAndInstanceInitializer(String template) {
        List<TestCase> list = new ArrayList<>();
        for (Modifier modifier : new Modifier[]{Modifier.EMPTY, Modifier.STATIC}) {
            list.addAll(generate(template, modifier.getString() + "{", "}"));
        }
        return list;
    }

    private Set<String> getFlags(Modifier modifier) {
        HashSet<String> set = new HashSet<>();
        if (modifier != Modifier.EMPTY) {
            set.add("ACC_" + modifier.getString().toUpperCase());
        }
        return set;
    }

    /**
     * Class represents part of sources which is inserted in other code.
     */
    private static class Data {
        public final String sources;
        public final Map<String, Set<String>> class2Flags;

        public Data(String sources, Map<String, Set<String>> class2Flags) {
            this.sources = sources;
            this.class2Flags = class2Flags;
        }
    }
}