blob: 2cae4528a8b528081d55669d85481c5f05fe5440 [file] [log] [blame]
/*******************************************************************************
* Copyright (c) 2019 Sebastian Zarnekow and others.
*
* This program and the accompanying materials
* are made available under the terms of the Eclipse Public License 2.0
* which accompanies this distribution, and is available at
* https://www.eclipse.org/legal/epl-2.0/
*
* SPDX-License-Identifier: EPL-2.0
*
* Contributors:
* Sebastian Zarnekow - initial API and implementation
*******************************************************************************/
package org.eclipse.jdt.core.tests.builder;
import static org.junit.Assert.assertArrayEquals;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.DataInputStream;
import java.io.DataOutputStream;
import java.io.IOException;
import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.Map;
import java.util.Set;
import org.eclipse.core.resources.IProject;
import org.eclipse.core.resources.IResource;
import org.eclipse.core.resources.IncrementalProjectBuilder;
import org.eclipse.core.resources.ResourcesPlugin;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.core.runtime.IPath;
import org.eclipse.core.runtime.Path;
import org.eclipse.jdt.core.IAccessRule;
import org.eclipse.jdt.core.IClasspathAttribute;
import org.eclipse.jdt.core.IClasspathEntry;
import org.eclipse.jdt.core.IJavaProject;
import org.eclipse.jdt.core.JavaCore;
import org.eclipse.jdt.core.JavaModelException;
import org.eclipse.jdt.core.compiler.CharOperation;
import org.eclipse.jdt.core.tests.util.Util;
import org.eclipse.jdt.internal.core.JavaModelManager;
import org.eclipse.jdt.internal.core.JavaModelManager.PerProjectInfo;
import org.eclipse.jdt.internal.core.builder.ClasspathLocation;
import org.eclipse.jdt.internal.core.builder.JavaBuilder;
import org.eclipse.jdt.internal.core.builder.ReferenceCollection;
import org.eclipse.jdt.internal.core.builder.State;
import junit.framework.Test;
public class StateTest extends BuilderTests {
public StateTest(String name) {
super(name);
}
public static Test suite() {
return buildTestSuite(StateTest.class);
}
public void testWriteAndReadState() throws JavaModelException, Exception {
IPath interfaceProjectPath = env.addProject("Interface"); //$NON-NLS-1$
env.addExternalJars(interfaceProjectPath, Util.getJavaClassLibs());
IPath implementationProjectPath = env.addProject("Implementation"); //$NON-NLS-1$
env.addExternalJars(implementationProjectPath, Util.getJavaClassLibs());
env.addClassFolder(implementationProjectPath, interfaceProjectPath, false);
env.addClass(interfaceProjectPath, "a", "Interfaze", //$NON-NLS-1$ //$NON-NLS-2$
"package a;\n" +
"public interface Interfaze {\n" +
" void callMe();\n" +
"}" //$NON-NLS-1$
);
env.addClass(interfaceProjectPath, "c", "Impl1", //$NON-NLS-1$ //$NON-NLS-2$
"package c;\n" +
"import a.Interfaze;\n" +
"public class Impl1 implements Interfaze {\n" +
" @Override\n" +
" public void callMe() {\n" +
" }\n" +
"}"
);
env.addClass(implementationProjectPath, "b", "Impl2", //$NON-NLS-1$ //$NON-NLS-2$
"package b;\n" +
"import a.Interfaze;\n" +
"public class Impl2 implements Interfaze {\n" +
" @Override\n" +
" public void callMe() {\n" +
" }\n" +
"}" //$NON-NLS-1$
);
fullBuild();
writeReadAndCompareReferences(interfaceProjectPath);
writeReadAndCompareReferences(implementationProjectPath);
}
public void testBug563546() throws JavaModelException, Exception {
IPath project = env.addProject("Bug563546"); //$NON-NLS-1$
env.addExternalJars(project, Util.getJavaClassLibs());
env.addClass(project, "a", "WithOther", //$NON-NLS-1$ //$NON-NLS-2$
"package a;\n" +
"class Other {\n" +
"}\n" +
"public class WithOther {\n" +
"}" //$NON-NLS-1$
);
fullBuild();
env.removePackage(project, "a");
incrementalBuild();
writeReadAndCompareReferences(project);
}
public void testBug567532() throws JavaModelException, Exception {
IPath project = env.addProject("Bug567532"); //$NON-NLS-1$
String[] classLibs = Util.getJavaClassLibs();
for (String jar : classLibs) {
env.addEntry(project,
JavaCore.newLibraryEntry(
new Path(jar),
null,
null,
new IAccessRule[0],
new IClasspathAttribute[] {
JavaCore.newClasspathAttribute(IClasspathAttribute.TEST, "true"),
JavaCore.newClasspathAttribute(IClasspathAttribute.ADD_EXPORTS,
"jdk.compiler/com.sun.tools.javac.util=ALL-UNNAMED") },
false));
}
env.addClass(project, "a", "WithOther", //$NON-NLS-1$ //$NON-NLS-2$
"package a;\n" +
"class Other {\n" +
"}\n" +
"public class WithOther {\n" +
"}" //$NON-NLS-1$
);
fullBuild();
env.removePackage(project, "a");
incrementalBuild();
writeReadAndCompareTestBinaryLocations(project);
}
public void testSelfAnnotatedJars() throws CoreException, IOException {
// derived from the same named test in ExternalAnnotation18Test:
IPath projectPath = env.addProject("PrjTest", "1.8"); //$NON-NLS-1$
IJavaProject project = JavaCore.create(ResourcesPlugin.getWorkspace().getRoot().getProject(projectPath.segment(0)));
project.setOption(JavaCore.CORE_JAVA_BUILD_EXTERNAL_ANNOTATIONS_FROM_ALL_LOCATIONS, JavaCore.ENABLED);
String projectLoc = project.getProject().getLocation().toString();
Util.createJar(new String[] {
"pgen/CGen.java",
"package pgen;\n" +
"public class CGen {\n" +
" public String get(String in) { return in; }\n" +
"}\n"
},
new String[] {
"pgen/CGen.eea",
"class pgen/CGen\n" +
"\n" +
"get\n" +
" (Ljava/lang/String;)Ljava/lang/String;\n" +
" (L1java/lang/String;)L1java/lang/String;\n"
},
projectLoc+"/lib/prj1.jar",
"1.8");
IClasspathEntry entry = JavaCore.newLibraryEntry(
new Path("/PrjTest/lib/prj1.jar"),
null/*access rules*/,
null,
false/*exported*/);
env.addEntry(project.getPath(), entry);
Util.createJar(new String[] {
"pgen2/CGen2.java",
"package pgen2;\n" +
"public class CGen2 {\n" +
" public String get2(Exception in) { return in.toString(); }\n" +
"}\n"
},
new String[] {
"pgen2/CGen2.eea",
"class pgen2/CGen2\n" +
"\n" +
"get2\n" +
" (Ljava/lang/Exception;)Ljava/lang/String;\n" +
" (L1java/lang/Exception;)L1java/lang/String;\n",
},
projectLoc+"/lib/prj2.jar",
"1.8");
entry = JavaCore.newLibraryEntry(
new Path("/PrjTest/lib/prj2.jar"),
null/*access rules*/,
null,
false/*exported*/);
env.addEntry(project.getPath(), entry);
project.getProject().refreshLocal(IResource.DEPTH_INFINITE, null);
env.addFolder(project.getPath(), "src/p");
env.addFile(project.getPath().append("src").append("p"), "Use.java",
"package p;\n" +
"import pgen.CGen;\n" +
"import pgen2.CGen2;\n" +
"import org.eclipse.jdt.annotation.NonNull;\n" +
"public class Use {\n" +
" public @NonNull String test(CGen c) {\n" +
" String s = c.get(null);\n" +
" return s;\n" +
" }\n" +
" public @NonNull String test2(CGen2 c) {\n" +
" String s = c.get2(null);\n" +
" return s;\n" +
" }\n" +
"}\n");
project.getProject().build(IncrementalProjectBuilder.FULL_BUILD, null);
writeReadAndCompareExternalAnnotationLocations(project.getProject());
}
private void writeReadAndCompareTestBinaryLocations(IPath projectPath)
throws JavaModelException, IOException, CoreException {
JavaModelManager javaModelManager = JavaModelManager.getJavaModelManager();
IProject project = env.getProject(projectPath);
PerProjectInfo info = javaModelManager.getPerProjectInfoCheckExistence(project);
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
State savedState = (State) info.savedState;
JavaBuilder.writeState(savedState, new DataOutputStream(outputStream));
byte[] bytes = outputStream.toByteArray();
State readState = JavaBuilder.readState(project, new DataInputStream(new ByteArrayInputStream(bytes)));
assertEqualBinaryLocations(savedState.testBinaryLocations, readState.testBinaryLocations);
}
private void assertEqualBinaryLocations(ClasspathLocation[] a,
ClasspathLocation[] b) {
assertEquals(a.length, b.length);
assertArrayEquals(a, b);
}
private void writeReadAndCompareExternalAnnotationLocations(IProject project)
throws JavaModelException, IOException, CoreException {
JavaModelManager javaModelManager = JavaModelManager.getJavaModelManager();
PerProjectInfo info = javaModelManager.getPerProjectInfoCheckExistence(project);
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
State savedState = (State) info.savedState;
JavaBuilder.writeState(savedState, new DataOutputStream(outputStream));
byte[] bytes = outputStream.toByteArray();
State readState = JavaBuilder.readState(project, new DataInputStream(new ByteArrayInputStream(bytes)));
assertArrayEquals(savedState.binaryLocations, readState.binaryLocations);
// beyond this point we know that both arrays have the same length
for (int i=0; i < savedState.binaryLocations.length; i++) {
assertTrue("comparing eea locations of "+savedState.binaryLocations[i], savedState.binaryLocations[i].externalAnnotationsEquals(readState.binaryLocations[i]));
}
}
private void writeReadAndCompareReferences(IPath projectPath)
throws JavaModelException, IOException, CoreException {
JavaModelManager javaModelManager = JavaModelManager.getJavaModelManager();
IProject project = env.getProject(projectPath);
PerProjectInfo info = javaModelManager.getPerProjectInfoCheckExistence(project);
ByteArrayOutputStream outputStream = new ByteArrayOutputStream();
State savedState = (State) info.savedState;
JavaBuilder.writeState(savedState, new DataOutputStream(outputStream));
byte[] bytes = outputStream.toByteArray();
State readState = JavaBuilder.readState(project, new DataInputStream(new ByteArrayInputStream(bytes)));
Map<String, ReferenceCollection> readReferences = readState.getReferences();
assertEqualLookupTables(savedState.getReferences(), readReferences);
assertEqualTypeLocators(savedState.typeLocators, readState.typeLocators);
}
private void assertEqualTypeLocators(Map<String, String> tl1, Map<String, String> tl2) {
assertEquals(tl1.size(), tl2.size());
assertEquals(tl1.toString(), tl2.toString());
}
private void assertEqualLookupTables(Map<String, ReferenceCollection> expectation, Map<String, ReferenceCollection> actual) {
assertEquals(expectation.size(), actual.size());
Set<String> expectedKeys = expectation.keySet();
for (String key : expectedKeys) {
ReferenceCollection actualReferenceCollection = actual.get(key);
ReferenceCollection expectedReferenceCollection = expectation.get(key);
assertEqualReferenceCollections(expectedReferenceCollection, actualReferenceCollection);
}
}
private void assertEqualReferenceCollections(ReferenceCollection expectedReferenceCollection,
ReferenceCollection actualReferenceCollection) {
{
char[][] expected = getSimpleNameReferences(expectedReferenceCollection);
char[][] actual = getSimpleNameReferences(actualReferenceCollection);
assertArrayEquals(toStringArray(expected), toStringArray(actual));
}
{
char[][] expected = getRootReferences(expectedReferenceCollection);
char[][] actual = getRootReferences(actualReferenceCollection);
assertArrayEquals(toStringArray(expected), toStringArray(actual));
}
{
char[][][] expected = getQualifiedNameReferences(expectedReferenceCollection);
char[][][] actual = getQualifiedNameReferences(actualReferenceCollection);
assertArrayEquals(toStringArray(expected), toStringArray(actual));
}
}
private static String[] toStringArray(char[][][] qualifiedNameReferences) {
return Arrays.stream(qualifiedNameReferences).map(a -> CharOperation.toString(a)).toArray(String[]::new);
}
private static String[] toStringArray(char[][] qualifiedNameReferences) {
return Arrays.stream(qualifiedNameReferences).map(a -> CharOperation.charToString(a)).toArray(String[]::new);
}
char[][][] getQualifiedNameReferences(ReferenceCollection collection) {
try {
Field fld = ReferenceCollection.class.getDeclaredField("qualifiedNameReferences");
fld.setAccessible(true);
return (char[][][]) fld.get(collection);
} catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) {
throw new RuntimeException(e);
}
}
char[][] getSimpleNameReferences(ReferenceCollection collection) {
try {
Field fld = ReferenceCollection.class.getDeclaredField("simpleNameReferences");
fld.setAccessible(true);
return (char[][]) fld.get(collection);
} catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) {
throw new RuntimeException(e);
}
}
char[][] getRootReferences(ReferenceCollection collection) {
try {
Field fld = ReferenceCollection.class.getDeclaredField("rootReferences");
fld.setAccessible(true);
return (char[][]) fld.get(collection);
} catch (NoSuchFieldException | SecurityException | IllegalArgumentException | IllegalAccessException e) {
throw new RuntimeException(e);
}
}
}