Chris Hegarty e11aec59a2 8071474: Better failure atomicity for default read object
Reviewed-by: plevart, coffeys
2015-06-03 15:30:44 +01:00

422 lines
18 KiB
Java

/*
* Copyright (c) 2015, 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 8071474
* @summary Better failure atomicity for default read object.
* @library /lib/testlibrary
* @build jdk.testlibrary.FileUtils
* @compile FailureAtomicity.java SerialRef.java
* @run main failureAtomicity.FailureAtomicity
*/
package failureAtomicity;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.ObjectStreamClass;
import java.io.UncheckedIOException;
import java.lang.reflect.Constructor;
import java.net.URL;
import java.net.URLClassLoader;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.function.BiConsumer;
import java.util.stream.Collectors;
import javax.tools.JavaCompiler;
import javax.tools.JavaFileObject;
import javax.tools.StandardJavaFileManager;
import javax.tools.StandardLocation;
import javax.tools.ToolProvider;
import jdk.testlibrary.FileUtils;
@SuppressWarnings("unchecked")
public class FailureAtomicity {
static final Path TEST_SRC = Paths.get(System.getProperty("test.src", "."));
static final Path TEST_CLASSES = Paths.get(System.getProperty("test.classes", "."));
static final Path fooTemplate = TEST_SRC.resolve("Foo.template");
static final Path barTemplate = TEST_SRC.resolve("Bar.template");
static final String[] PKGS = { "a.b.c", "x.y.z" };
public static void main(String[] args) throws Exception {
test_Foo();
test_BadFoo(); // 'Bad' => incompatible type; cannot be "fully" deserialized
test_FooWithReadObject();
test_BadFooWithReadObject();
test_Foo_Bar();
test_Foo_BadBar();
test_BadFoo_Bar();
test_BadFoo_BadBar();
test_Foo_BarWithReadObject();
test_Foo_BadBarWithReadObject();
test_BadFoo_BarWithReadObject();
test_BadFoo_BadBarWithReadObject();
test_FooWithReadObject_Bar();
test_FooWithReadObject_BadBar();
test_BadFooWithReadObject_Bar();
test_BadFooWithReadObject_BadBar();
}
static final BiConsumer<Object,Object> FOO_FIELDS_EQUAL = (a,b) -> {
try {
int aPrim = a.getClass().getField("fooPrim").getInt(a);
int bPrim = b.getClass().getField("fooPrim").getInt(b);
if (aPrim != bPrim)
throw new AssertionError("Not equal: (" + aPrim + "!=" + bPrim
+ "), in [" + a + "] [" + b + "]");
Object aRef = a.getClass().getField("fooRef").get(a);
Object bRef = b.getClass().getField("fooRef").get(b);
if (!aRef.equals(bRef))
throw new RuntimeException("Not equal: (" + aRef + "!=" + bRef
+ "), in [" + a + "] [" + b + "]");
} catch (NoSuchFieldException | IllegalAccessException x) {
throw new InternalError(x);
}
};
static final BiConsumer<Object,Object> FOO_FIELDS_DEFAULT = (ignore,b) -> {
try {
int aPrim = b.getClass().getField("fooPrim").getInt(b);
if (aPrim != 0)
throw new AssertionError("Expected 0, got:" + aPrim
+ ", in [" + b + "]");
Object aRef = b.getClass().getField("fooRef").get(b);
if (aRef != null)
throw new RuntimeException("Expected null, got:" + aRef
+ ", in [" + b + "]");
} catch (NoSuchFieldException | IllegalAccessException x) {
throw new InternalError(x);
}
};
static final BiConsumer<Object,Object> BAR_FIELDS_EQUAL = (a,b) -> {
try {
long aPrim = a.getClass().getField("barPrim").getLong(a);
long bPrim = b.getClass().getField("barPrim").getLong(b);
if (aPrim != bPrim)
throw new AssertionError("Not equal: (" + aPrim + "!=" + bPrim
+ "), in [" + a + "] [" + b + "]");
Object aRef = a.getClass().getField("barRef").get(a);
Object bRef = b.getClass().getField("barRef").get(b);
if (!aRef.equals(bRef))
throw new RuntimeException("Not equal: (" + aRef + "!=" + bRef
+ "), in [" + a + "] [" + b + "]");
} catch (NoSuchFieldException | IllegalAccessException x) {
throw new InternalError(x);
}
};
static final BiConsumer<Object,Object> BAR_FIELDS_DEFAULT = (ignore,b) -> {
try {
long aPrim = b.getClass().getField("barPrim").getLong(b);
if (aPrim != 0L)
throw new AssertionError("Expected 0, got:" + aPrim
+ ", in [" + b + "]");
Object aRef = b.getClass().getField("barRef").get(b);
if (aRef != null)
throw new RuntimeException("Expected null, got:" + aRef
+ ", in [" + b + "]");
} catch (NoSuchFieldException | IllegalAccessException x) {
throw new InternalError(x);
}
};
static void test_Foo() {
testFoo("Foo", "String", false, false, FOO_FIELDS_EQUAL); }
static void test_BadFoo() {
testFoo("BadFoo", "byte[]", true, false, FOO_FIELDS_DEFAULT); }
static void test_FooWithReadObject() {
testFoo("FooWithReadObject", "String", false, true, FOO_FIELDS_EQUAL); }
static void test_BadFooWithReadObject() {
testFoo("BadFooWithReadObject", "byte[]", true, true, FOO_FIELDS_DEFAULT); }
static void testFoo(String testName, String xyzZebraType,
boolean expectCCE, boolean withReadObject,
BiConsumer<Object,Object>... resultCheckers) {
System.out.println("\nTesting " + testName);
try {
Path testRoot = testDir(testName);
Path srcRoot = Files.createDirectory(testRoot.resolve("src"));
List<Path> srcFiles = new ArrayList<>();
srcFiles.add(createSrc(PKGS[0], fooTemplate, srcRoot, "String", withReadObject));
srcFiles.add(createSrc(PKGS[1], fooTemplate, srcRoot, xyzZebraType, withReadObject));
Path build = Files.createDirectory(testRoot.resolve("build"));
javac(build, srcFiles);
URLClassLoader loader = new URLClassLoader(new URL[]{ build.toUri().toURL() },
FailureAtomicity.class.getClassLoader());
Class<?> fooClass = Class.forName(PKGS[0] + ".Foo", true, loader);
Constructor<?> ctr = fooClass.getConstructor(
new Class<?>[]{int.class, String.class, String.class});
Object abcFoo = ctr.newInstance(5, "chegar", "zebra");
try {
toOtherPkgInstance(abcFoo, loader);
if (expectCCE)
throw new AssertionError("Expected CCE not thrown");
} catch (ClassCastException e) {
if (!expectCCE)
throw new AssertionError("UnExpected CCE: " + e);
}
Object deserialInstance = failureAtomicity.SerialRef.obj;
System.out.println("abcFoo: " + abcFoo);
System.out.println("deserialInstance: " + deserialInstance);
for (BiConsumer<Object, Object> rc : resultCheckers)
rc.accept(abcFoo, deserialInstance);
} catch (IOException x) {
throw new UncheckedIOException(x);
} catch (ReflectiveOperationException x) {
throw new InternalError(x);
}
}
static void test_Foo_Bar() {
testFooBar("Foo_Bar", "String", "String", false, false, false,
FOO_FIELDS_EQUAL, BAR_FIELDS_EQUAL);
}
static void test_Foo_BadBar() {
testFooBar("Foo_BadBar", "String", "byte[]", true, false, false,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void test_BadFoo_Bar() {
testFooBar("BadFoo_Bar", "byte[]", "String", true, false, false,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void test_BadFoo_BadBar() {
testFooBar("BadFoo_BadBar", "byte[]", "byte[]", true, false, false,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void test_Foo_BarWithReadObject() {
testFooBar("Foo_BarWithReadObject", "String", "String", false, false, true,
FOO_FIELDS_EQUAL, BAR_FIELDS_EQUAL);
}
static void test_Foo_BadBarWithReadObject() {
testFooBar("Foo_BadBarWithReadObject", "String", "byte[]", true, false, true,
FOO_FIELDS_EQUAL, BAR_FIELDS_DEFAULT);
}
static void test_BadFoo_BarWithReadObject() {
testFooBar("BadFoo_BarWithReadObject", "byte[]", "String", true, false, true,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void test_BadFoo_BadBarWithReadObject() {
testFooBar("BadFoo_BadBarWithReadObject", "byte[]", "byte[]", true, false, true,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void test_FooWithReadObject_Bar() {
testFooBar("FooWithReadObject_Bar", "String", "String", false, true, false,
FOO_FIELDS_EQUAL, BAR_FIELDS_EQUAL);
}
static void test_FooWithReadObject_BadBar() {
testFooBar("FooWithReadObject_BadBar", "String", "byte[]", true, true, false,
FOO_FIELDS_EQUAL, BAR_FIELDS_DEFAULT);
}
static void test_BadFooWithReadObject_Bar() {
testFooBar("BadFooWithReadObject_Bar", "byte[]", "String", true, true, false,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void test_BadFooWithReadObject_BadBar() {
testFooBar("BadFooWithReadObject_BadBar", "byte[]", "byte[]", true, true, false,
FOO_FIELDS_DEFAULT, BAR_FIELDS_DEFAULT);
}
static void testFooBar(String testName, String xyzFooZebraType,
String xyzBarZebraType, boolean expectCCE,
boolean fooWithReadObject, boolean barWithReadObject,
BiConsumer<Object,Object>... resultCheckers) {
System.out.println("\nTesting " + testName);
try {
Path testRoot = testDir(testName);
Path srcRoot = Files.createDirectory(testRoot.resolve("src"));
List<Path> srcFiles = new ArrayList<>();
srcFiles.add(createSrc(PKGS[0], fooTemplate, srcRoot, "String",
fooWithReadObject, "String"));
srcFiles.add(createSrc(PKGS[1], fooTemplate, srcRoot, xyzFooZebraType,
fooWithReadObject, xyzFooZebraType));
srcFiles.add(createSrc(PKGS[0], barTemplate, srcRoot, "String",
barWithReadObject, "String"));
srcFiles.add(createSrc(PKGS[1], barTemplate, srcRoot, xyzBarZebraType,
barWithReadObject, xyzFooZebraType));
Path build = Files.createDirectory(testRoot.resolve("build"));
javac(build, srcFiles);
URLClassLoader loader = new URLClassLoader(new URL[]{ build.toUri().toURL() },
FailureAtomicity.class.getClassLoader());
Class<?> fooClass = Class.forName(PKGS[0] + ".Bar", true, loader);
Constructor<?> ctr = fooClass.getConstructor(
new Class<?>[]{int.class, String.class, String.class,
long.class, String.class, String.class});
Object abcBar = ctr.newInstance( 5, "chegar", "zebraFoo", 111L, "aBar", "zebraBar");
try {
toOtherPkgInstance(abcBar, loader);
if (expectCCE)
throw new AssertionError("Expected CCE not thrown");
} catch (ClassCastException e) {
if (!expectCCE)
throw new AssertionError("UnExpected CCE: " + e);
}
Object deserialInstance = failureAtomicity.SerialRef.obj;
System.out.println("abcBar: " + abcBar);
System.out.println("deserialInstance: " + deserialInstance);
for (BiConsumer<Object, Object> rc : resultCheckers)
rc.accept(abcBar, deserialInstance);
} catch (IOException x) {
throw new UncheckedIOException(x);
} catch (ReflectiveOperationException x) {
throw new InternalError(x);
}
}
static Path testDir(String name) throws IOException {
Path testRoot = Paths.get("FailureAtomicity-" + name);
if (Files.exists(testRoot))
FileUtils.deleteFileTreeWithRetry(testRoot);
Files.createDirectory(testRoot);
return testRoot;
}
static String platformPath(String p) { return p.replace("/", File.separator); }
static String binaryName(String name) { return name.replace(".", "/"); }
static String condRemove(String line, String pattern, boolean hasReadObject) {
if (hasReadObject) { return line.replaceAll(pattern, ""); }
else { return line; }
}
static String condReplace(String line, String... zebraFooType) {
if (zebraFooType.length == 1) {
return line.replaceAll("\\$foo_zebra_type", zebraFooType[0]);
} else { return line; }
}
static String nameFromTemplate(Path template) {
return template.getFileName().toString().replaceAll(".template", "");
}
static Path createSrc(String pkg, Path srcTemplate, Path srcRoot,
String zebraType, boolean hasReadObject,
String... zebraFooType)
throws IOException
{
Path srcDst = srcRoot.resolve(platformPath(binaryName(pkg)));
Files.createDirectories(srcDst);
Path srcFile = srcDst.resolve(nameFromTemplate(srcTemplate) + ".java");
List<String> lines = Files.lines(srcTemplate)
.map(s -> s.replaceAll("\\$package", pkg))
.map(s -> s.replaceAll("\\$zebra_type", zebraType))
.map(s -> condReplace(s, zebraFooType))
.map(s -> condRemove(s, "//\\$has_readObject", hasReadObject))
.collect(Collectors.toList());
Files.write(srcFile, lines);
return srcFile;
}
static void javac(Path dest, List<Path> sourceFiles) throws IOException {
JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
try (StandardJavaFileManager fileManager =
compiler.getStandardFileManager(null, null, null)) {
List<File> files = sourceFiles.stream()
.map(p -> p.toFile())
.collect(Collectors.toList());
Iterable<? extends JavaFileObject> compilationUnits =
fileManager.getJavaFileObjectsFromFiles(files);
fileManager.setLocation(StandardLocation.CLASS_OUTPUT,
Arrays.asList(dest.toFile()));
fileManager.setLocation(StandardLocation.CLASS_PATH,
Arrays.asList(TEST_CLASSES.toFile()));
JavaCompiler.CompilationTask task = compiler
.getTask(null, fileManager, null, null, null, compilationUnits);
boolean passed = task.call();
if (!passed)
throw new RuntimeException("Error compiling " + files);
}
}
static Object toOtherPkgInstance(Object obj, ClassLoader loader)
throws IOException, ClassNotFoundException
{
byte[] bytes = serialize(obj);
bytes = replacePkg(bytes);
return deserialize(bytes, loader);
}
@SuppressWarnings("deprecation")
static byte[] replacePkg(byte[] bytes) {
String str = new String(bytes, 0);
str = str.replaceAll(PKGS[0], PKGS[1]);
str.getBytes(0, bytes.length, bytes, 0);
return bytes;
}
static byte[] serialize(Object obj) throws IOException {
try (ByteArrayOutputStream baos = new ByteArrayOutputStream();
ObjectOutputStream out = new ObjectOutputStream(baos);) {
out.writeObject(obj);
return baos.toByteArray();
}
}
static Object deserialize(byte[] data, ClassLoader l)
throws IOException, ClassNotFoundException
{
return new WithLoaderObjectInputStream(new ByteArrayInputStream(data), l)
.readObject();
}
static class WithLoaderObjectInputStream extends ObjectInputStream {
final ClassLoader loader;
WithLoaderObjectInputStream(InputStream is, ClassLoader loader)
throws IOException
{
super(is);
this.loader = loader;
}
@Override
protected Class<?> resolveClass(ObjectStreamClass desc)
throws IOException, ClassNotFoundException {
try {
return super.resolveClass(desc);
} catch (ClassNotFoundException x) {
String name = desc.getName();
return Class.forName(name, false, loader);
}
}
}
}