提交 766288ac authored 作者: andrei's avatar andrei

TestOutOfMemory improvements: propagate OOM from background thread

上级 bc763b82
...@@ -13,6 +13,7 @@ import java.sql.SQLException; ...@@ -13,6 +13,7 @@ import java.sql.SQLException;
import java.sql.Statement; import java.sql.Statement;
import java.util.Map; import java.util.Map;
import java.util.Random; import java.util.Random;
import java.util.concurrent.atomic.AtomicReference;
import org.h2.api.ErrorCode; import org.h2.api.ErrorCode;
import org.h2.mvstore.MVStore; import org.h2.mvstore.MVStore;
import org.h2.store.fs.FilePath; import org.h2.store.fs.FilePath;
...@@ -37,10 +38,6 @@ public class TestOutOfMemory extends TestBase { ...@@ -37,10 +38,6 @@ public class TestOutOfMemory extends TestBase {
@Override @Override
public void test() throws SQLException, InterruptedException { public void test() throws SQLException, InterruptedException {
if (config.travis) {
// fails regularly under Travis, not sure why
return;
}
if (config.vmlens) { if (config.vmlens) {
// running out of memory will cause the vmlens agent to stop working // running out of memory will cause the vmlens agent to stop working
return; return;
...@@ -60,7 +57,16 @@ public class TestOutOfMemory extends TestBase { ...@@ -60,7 +57,16 @@ public class TestOutOfMemory extends TestBase {
private void testMVStoreUsingInMemoryFileSystem() { private void testMVStoreUsingInMemoryFileSystem() {
FilePath.register(new FilePathMem()); FilePath.register(new FilePathMem());
String fileName = "memFS:" + getTestName(); String fileName = "memFS:" + getTestName();
MVStore store = MVStore.open(fileName); final AtomicReference<Throwable> exRef = new AtomicReference<>();
MVStore store = new MVStore.Builder()
.fileName(fileName)
.backgroundExceptionHandler(new Thread.UncaughtExceptionHandler() {
@Override
public void uncaughtException(Thread t, Throwable e) {
exRef.compareAndSet(null, e);
}
})
.open();
try { try {
Map<Integer, byte[]> map = store.openMap("test"); Map<Integer, byte[]> map = store.openMap("test");
Random r = new Random(1); Random r = new Random(1);
...@@ -70,10 +76,11 @@ public class TestOutOfMemory extends TestBase { ...@@ -70,10 +76,11 @@ public class TestOutOfMemory extends TestBase {
r.nextBytes(data); r.nextBytes(data);
map.put(i, data); map.put(i, data);
} }
Throwable throwable = exRef.get();
if(throwable instanceof OutOfMemoryError) throw (OutOfMemoryError)throwable;
if(throwable instanceof IllegalStateException) throw (IllegalStateException)throwable;
fail(); fail();
} catch (OutOfMemoryError e) { } catch (OutOfMemoryError | IllegalStateException e) {
// expected
} catch (IllegalStateException e) {
// expected // expected
} }
try { try {
...@@ -83,7 +90,7 @@ public class TestOutOfMemory extends TestBase { ...@@ -83,7 +90,7 @@ public class TestOutOfMemory extends TestBase {
} }
store.closeImmediately(); store.closeImmediately();
store = MVStore.open(fileName); store = MVStore.open(fileName);
map = store.openMap("test"); store.openMap("test");
store.close(); store.close();
} finally { } finally {
// just in case, otherwise if this test suffers a spurious failure, // just in case, otherwise if this test suffers a spurious failure,
...@@ -95,26 +102,30 @@ public class TestOutOfMemory extends TestBase { ...@@ -95,26 +102,30 @@ public class TestOutOfMemory extends TestBase {
private void testDatabaseUsingInMemoryFileSystem() throws SQLException, InterruptedException { private void testDatabaseUsingInMemoryFileSystem() throws SQLException, InterruptedException {
String filename = "memFS:" + getTestName(); String filename = "memFS:" + getTestName();
String url = "jdbc:h2:" + filename; String url = "jdbc:h2:" + filename + "/test";
try {
Connection conn = DriverManager.getConnection(url); Connection conn = DriverManager.getConnection(url);
Statement stat = conn.createStatement(); Statement stat = conn.createStatement();
try { try {
stat.execute("create table test(id int, name varchar) as " + stat.execute("create table test(id int, name varchar) as " +
"select x, space(10000000) from system_range(1, 1000)"); "select x, space(10000000+x) from system_range(1, 1000)");
fail(); fail();
} catch (SQLException e) { } catch (SQLException e) {
int err = e.getErrorCode(); assertTrue("Unexpected error code: " + e.getErrorCode(),
assertTrue(e.getMessage(), err == ErrorCode.GENERAL_ERROR_1 ErrorCode.OUT_OF_MEMORY == e.getErrorCode() ||
|| err == ErrorCode.OUT_OF_MEMORY); ErrorCode.FILE_CORRUPTED_1 == e.getErrorCode() ||
ErrorCode.DATABASE_IS_CLOSED == e.getErrorCode() ||
ErrorCode.GENERAL_ERROR_1 == e.getErrorCode());
} }
try { try {
conn.close(); conn.close();
fail(); fail();
} catch (SQLException e) { } catch (SQLException e) {
int err = e.getErrorCode(); assertTrue("Unexpected error code: " + e.getErrorCode(),
assertTrue(e.getMessage(), err == ErrorCode.GENERAL_ERROR_1 ErrorCode.OUT_OF_MEMORY == e.getErrorCode() ||
|| err == ErrorCode.OUT_OF_MEMORY ErrorCode.FILE_CORRUPTED_1 == e.getErrorCode() ||
|| err == ErrorCode.DATABASE_IS_CLOSED); ErrorCode.DATABASE_IS_CLOSED == e.getErrorCode() ||
ErrorCode.GENERAL_ERROR_1 == e.getErrorCode());
} }
for (int i = 0; i < 5; i++) { for (int i = 0; i < 5; i++) {
System.gc(); System.gc();
...@@ -122,10 +133,12 @@ public class TestOutOfMemory extends TestBase { ...@@ -122,10 +133,12 @@ public class TestOutOfMemory extends TestBase {
} }
conn = DriverManager.getConnection(url); conn = DriverManager.getConnection(url);
stat = conn.createStatement(); stat = conn.createStatement();
stat.execute("select 1"); stat.execute("SELECT 1");
conn.close(); conn.close();
} finally {
// release the static data this test generates // release the static data this test generates
FileUtils.delete(filename); FileUtils.deleteRecursive(filename, true);
}
} }
private void testUpdateWhenNearlyOutOfMemory() throws SQLException, InterruptedException { private void testUpdateWhenNearlyOutOfMemory() throws SQLException, InterruptedException {
...@@ -160,7 +173,7 @@ public class TestOutOfMemory extends TestBase { ...@@ -160,7 +173,7 @@ public class TestOutOfMemory extends TestBase {
} catch (OutOfMemoryError e) { } catch (OutOfMemoryError e) {
freeMemory(); freeMemory();
// out of memory not detected // out of memory not detected
throw (Error) new AssertionError("Out of memory not detected").initCause(e); throw new AssertionError("Out of memory not detected", e);
} finally { } finally {
freeMemory(); freeMemory();
if (conn != null) { if (conn != null) {
......
Markdown 格式
0%
您添加了 0 到此讨论。请谨慎行事。
请先完成此评论的编辑!
注册 或者 后发表评论