f1c7afcc3f
8306572: Implementation of Scoped Values (Preview) Co-authored-by: Alan Bateman <alanb@openjdk.org> Co-authored-by: Andrew Haley <aph@openjdk.org> Reviewed-by: psandoz, dfuchs, mchung
562 lines
19 KiB
Java
562 lines
19 KiB
Java
/*
|
|
* Copyright (c) 2021, 2023, 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
|
|
* @summary Test ScopedValue API
|
|
* @enablePreview
|
|
* @run junit ScopedValueAPI
|
|
*/
|
|
|
|
import java.util.NoSuchElementException;
|
|
import java.util.concurrent.Callable;
|
|
import java.util.concurrent.ExecutionException;
|
|
import java.util.concurrent.Executors;
|
|
import java.util.concurrent.ThreadFactory;
|
|
import java.util.function.Supplier;
|
|
import java.util.stream.Stream;
|
|
|
|
import org.junit.jupiter.api.Test;
|
|
import org.junit.jupiter.params.ParameterizedTest;
|
|
import org.junit.jupiter.params.provider.MethodSource;
|
|
import static org.junit.jupiter.api.Assertions.*;
|
|
|
|
class ScopedValueAPI {
|
|
|
|
private static Stream<ThreadFactory> factories() {
|
|
return Stream.of(Thread.ofPlatform().factory(), Thread.ofVirtual().factory());
|
|
}
|
|
|
|
/**
|
|
* Test that the run method is invoked.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testRun(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
class Box { static boolean executed; }
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
ScopedValue.runWhere(name, "duke", () -> { Box.executed = true; });
|
|
assertTrue(Box.executed);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test the run method throwing an exception.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testRunThrows(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
class FooException extends RuntimeException { }
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
Runnable op = () -> { throw new FooException(); };
|
|
assertThrows(FooException.class, () -> ScopedValue.runWhere(name, "duke", op));
|
|
assertFalse(name.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that the call method is invoked.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testCall(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
String result = ScopedValue.callWhere(name, "duke", name::get);
|
|
assertEquals("duke", result);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test that the get method is invoked.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testGetWhere(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
String result = ScopedValue.getWhere(name, "duke", (Supplier<String>)(name::get));
|
|
assertEquals("duke", result);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test the call method throwing an exception.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testCallThrows(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
class FooException extends RuntimeException { }
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
Callable<Void> op = () -> { throw new FooException(); };
|
|
assertThrows(FooException.class, () -> ScopedValue.callWhere(name, "duke", op));
|
|
assertFalse(name.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test the get(Supplier) method throwing an exception.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testGetThrows(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
class FooException extends RuntimeException { }
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
Supplier<Void> op = () -> { throw new FooException(); };
|
|
assertThrows(FooException.class, () -> ScopedValue.getWhere(name, "duke", op));
|
|
assertFalse(name.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test get method.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testGet(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name1 = ScopedValue.newInstance();
|
|
ScopedValue<String> name2 = ScopedValue.newInstance();
|
|
assertThrows(NoSuchElementException.class, name1::get);
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
|
|
// run
|
|
ScopedValue.runWhere(name1, "duke", () -> {
|
|
assertEquals("duke", name1.get());
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
|
|
});
|
|
assertThrows(NoSuchElementException.class, name1::get);
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
|
|
// call
|
|
ScopedValue.callWhere(name1, "duke", () -> {
|
|
assertEquals("duke", name1.get());
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
return null;
|
|
});
|
|
assertThrows(NoSuchElementException.class, name1::get);
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
|
|
// get
|
|
ScopedValue.getWhere(name1, "duke", () -> {
|
|
assertEquals("duke", name1.get());
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
return null;
|
|
});
|
|
assertThrows(NoSuchElementException.class, name1::get);
|
|
assertThrows(NoSuchElementException.class, name2::get);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test isBound method.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testIsBound(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name1 = ScopedValue.newInstance();
|
|
ScopedValue<String> name2 = ScopedValue.newInstance();
|
|
assertFalse(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
|
|
// run
|
|
ScopedValue.runWhere(name1, "duke", () -> {
|
|
assertTrue(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
});
|
|
assertFalse(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
|
|
// call
|
|
ScopedValue.callWhere(name1, "duke", () -> {
|
|
assertTrue(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
return null;
|
|
});
|
|
assertFalse(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
|
|
// call
|
|
ScopedValue.callWhere(name1, "duke", () -> {
|
|
assertTrue(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
return null;
|
|
});
|
|
assertFalse(name1.isBound());
|
|
assertFalse(name2.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test orElse method.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testOrElse(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
assertNull(name.orElse(null));
|
|
assertEquals("default", name.orElse("default"));
|
|
|
|
// run
|
|
ScopedValue.runWhere(name, "duke", () -> {
|
|
assertEquals("duke", name.orElse(null));
|
|
assertEquals("duke", name.orElse("default"));
|
|
});
|
|
|
|
// call
|
|
ScopedValue.callWhere(name, "duke", () -> {
|
|
assertEquals("duke", name.orElse(null));
|
|
assertEquals("duke", name.orElse("default"));
|
|
return null;
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test orElseThrow method.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testOrElseThrow(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
class FooException extends RuntimeException { }
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
assertThrows(FooException.class, () -> name.orElseThrow(FooException::new));
|
|
|
|
// run
|
|
ScopedValue.runWhere(name, "duke", () -> {
|
|
assertEquals("duke", name.orElseThrow(FooException::new));
|
|
});
|
|
|
|
// call
|
|
ScopedValue.callWhere(name, "duke", () -> {
|
|
assertEquals("duke", name.orElseThrow(FooException::new));
|
|
return null;
|
|
});
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test two bindings.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testTwoBindings(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
ScopedValue<Integer> age = ScopedValue.newInstance();
|
|
|
|
// run
|
|
ScopedValue.where(name, "duke").where(age, 100).run(() -> {
|
|
assertTrue(name.isBound());
|
|
assertTrue(age.isBound());
|
|
assertEquals("duke", name.get());
|
|
assertEquals(100, (int) age.get());
|
|
});
|
|
assertFalse(name.isBound());
|
|
assertFalse(age.isBound());
|
|
|
|
// call
|
|
ScopedValue.where(name, "duke").where(age, 100).call(() -> {
|
|
assertTrue(name.isBound());
|
|
assertTrue(age.isBound());
|
|
assertEquals("duke", name.get());
|
|
assertEquals(100, (int) age.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
assertFalse(age.isBound());
|
|
|
|
// get
|
|
ScopedValue.where(name, "duke").where(age, 100).get(() -> {
|
|
assertTrue(name.isBound());
|
|
assertTrue(age.isBound());
|
|
assertEquals("duke", name.get());
|
|
assertEquals(100, (int) age.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
assertFalse(age.isBound());
|
|
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test rebinding.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testRebinding(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
|
|
// run
|
|
ScopedValue.runWhere(name, "duke", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
|
|
ScopedValue.runWhere(name, "duchess", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duchess", name.get());
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
});
|
|
assertFalse(name.isBound());
|
|
|
|
// call
|
|
ScopedValue.callWhere(name, "duke", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
|
|
ScopedValue.callWhere(name, "duchess", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duchess", name.get());
|
|
return null;
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
|
|
// get
|
|
ScopedValue.getWhere(name, "duke", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
|
|
ScopedValue.where(name, "duchess").get(() -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duchess", name.get());
|
|
return null;
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test rebinding from null vaue to another value.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testRebindingFromNull(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
|
|
// run
|
|
ScopedValue.runWhere(name, null, () -> {
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
|
|
ScopedValue.runWhere(name, "duchess", () -> {
|
|
assertTrue(name.isBound());
|
|
assertTrue("duchess".equals(name.get()));
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
});
|
|
assertFalse(name.isBound());
|
|
|
|
// call
|
|
ScopedValue.callWhere(name, null, () -> {
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
|
|
ScopedValue.callWhere(name, "duchess", () -> {
|
|
assertTrue(name.isBound());
|
|
assertTrue("duchess".equals(name.get()));
|
|
return null;
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
|
|
// getWhere
|
|
ScopedValue.getWhere(name, null, () -> {
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
|
|
ScopedValue.getWhere(name, "duchess", () -> {
|
|
assertTrue(name.isBound());
|
|
assertTrue("duchess".equals(name.get()));
|
|
return null;
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test rebinding to null value.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testRebindingToNull(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
|
|
// run
|
|
ScopedValue.runWhere(name, "duke", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
|
|
ScopedValue.runWhere(name, null, () -> {
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
});
|
|
assertFalse(name.isBound());
|
|
|
|
// call
|
|
ScopedValue.callWhere(name, "duke", () -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
|
|
ScopedValue.callWhere(name, null, () -> {
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
return null;
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
|
|
// get
|
|
ScopedValue.where(name, "duke").get(() -> {
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
|
|
ScopedValue.where(name, null).get(() -> {
|
|
assertTrue(name.isBound());
|
|
assertNull(name.get());
|
|
return null;
|
|
});
|
|
|
|
assertTrue(name.isBound());
|
|
assertEquals("duke", name.get());
|
|
return null;
|
|
});
|
|
assertFalse(name.isBound());
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test Carrier.get.
|
|
*/
|
|
@ParameterizedTest
|
|
@MethodSource("factories")
|
|
void testCarrierGet(ThreadFactory factory) throws Exception {
|
|
test(factory, () -> {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
ScopedValue<Integer> age = ScopedValue.newInstance();
|
|
|
|
// one scoped value
|
|
var carrier1 = ScopedValue.where(name, "duke");
|
|
assertEquals("duke", carrier1.get(name));
|
|
assertThrows(NoSuchElementException.class, () -> carrier1.get(age));
|
|
|
|
// two scoped values
|
|
var carrier2 = carrier1.where(age, 20);
|
|
assertEquals("duke", carrier2.get(name));
|
|
assertEquals(20, (int) carrier2.get(age));
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Test NullPointerException.
|
|
*/
|
|
@Test
|
|
void testNullPointerException() {
|
|
ScopedValue<String> name = ScopedValue.newInstance();
|
|
|
|
assertThrows(NullPointerException.class, () -> ScopedValue.where(null, "value"));
|
|
assertThrows(NullPointerException.class, () -> ScopedValue.runWhere(null, "value", () -> { }));
|
|
assertThrows(NullPointerException.class, () -> ScopedValue.getWhere(null, "value", () -> null));
|
|
|
|
assertThrows(NullPointerException.class, () -> name.orElseThrow(null));
|
|
|
|
var carrier = ScopedValue.where(name, "duke");
|
|
assertThrows(NullPointerException.class, () -> carrier.where(null, "value"));
|
|
assertThrows(NullPointerException.class, () -> carrier.get((ScopedValue<?>)null));
|
|
assertThrows(NullPointerException.class, () -> carrier.get((Supplier<?>)null));
|
|
assertThrows(NullPointerException.class, () -> carrier.run(null));
|
|
assertThrows(NullPointerException.class, () -> carrier.call(null));
|
|
}
|
|
|
|
@FunctionalInterface
|
|
private interface ThrowingRunnable {
|
|
void run() throws Exception;
|
|
}
|
|
|
|
/**
|
|
* Run the given task in a thread created with the given thread factory.
|
|
* @throws Exception if the task throws an exception
|
|
*/
|
|
private static void test(ThreadFactory factory, ThrowingRunnable task) throws Exception {
|
|
try (var executor = Executors.newThreadPerTaskExecutor(factory)) {
|
|
var future = executor.submit(() -> {
|
|
task.run();
|
|
return null;
|
|
});
|
|
try {
|
|
future.get();
|
|
} catch (ExecutionException ee) {
|
|
Throwable cause = ee.getCause();
|
|
if (cause instanceof Exception e)
|
|
throw e;
|
|
if (cause instanceof Error e)
|
|
throw e;
|
|
throw new RuntimeException(cause);
|
|
}
|
|
}
|
|
}
|
|
}
|