Java 11 migrate remaining p (#1122)
* Moves partial-response to Java 11 * Moves pipeline to Java 11 * Moves poison-pill to Java 11 * Moves priority-queue to Java 11 * Moves private-class-data to Java 11 * Moves producer-consumer to Java 11 * Moves promise to Java 11 * Moves property to Java 11 * Moves prototype to Java 11 * Moves proxy to Java 11 * Corrects checkstyle errors * Fixes build for pipeline pattern
This commit is contained in:
committed by
Ilkka Seppälä
parent
1401accb4f
commit
428efc7d53
@@ -29,19 +29,18 @@ import java.util.concurrent.CountDownLatch;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.ExecutorService;
|
||||
import java.util.concurrent.Executors;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
/**
|
||||
* The Promise object is used for asynchronous computations. A Promise represents an operation
|
||||
* that hasn't completed yet, but is expected in the future.
|
||||
* The Promise object is used for asynchronous computations. A Promise represents an operation that
|
||||
* hasn't completed yet, but is expected in the future.
|
||||
*
|
||||
* <p>A Promise represents a proxy for a value not necessarily known when the promise is created. It
|
||||
* allows you to associate dependent promises to an asynchronous action's eventual success value or
|
||||
* failure reason. This lets asynchronous methods return values like synchronous methods: instead
|
||||
* of the final value, the asynchronous method returns a promise of having a value at some point
|
||||
* in the future.
|
||||
* <p>A Promise represents a proxy for a value not necessarily known when the promise is created.
|
||||
* It allows you to associate dependent promises to an asynchronous action's eventual success value
|
||||
* or failure reason. This lets asynchronous methods return values like synchronous methods: instead
|
||||
* of the final value, the asynchronous method returns a promise of having a value at some point in
|
||||
* the future.
|
||||
*
|
||||
* <p>Promises provide a few advantages over callback objects:
|
||||
* <ul>
|
||||
@@ -58,14 +57,15 @@ import org.slf4j.LoggerFactory;
|
||||
* a file download promise, then a promise of character frequency, then a promise of lowest
|
||||
* frequency character which is finally consumed and result is printed on console.
|
||||
* </ul>
|
||||
*
|
||||
*
|
||||
* @see CompletableFuture
|
||||
*/
|
||||
public class App {
|
||||
|
||||
private static final Logger LOGGER = LoggerFactory.getLogger(App.class);
|
||||
|
||||
private static final String DEFAULT_URL = "https://raw.githubusercontent.com/iluwatar/java-design-patterns/master/promise/README.md";
|
||||
private static final String DEFAULT_URL =
|
||||
"https://raw.githubusercontent.com/iluwatar/java-design-patterns/master/promise/README.md";
|
||||
private final ExecutorService executor;
|
||||
private final CountDownLatch stopLatch;
|
||||
|
||||
@@ -76,12 +76,13 @@ public class App {
|
||||
|
||||
/**
|
||||
* Program entry point.
|
||||
*
|
||||
* @param args arguments
|
||||
* @throws InterruptedException if main thread is interrupted.
|
||||
* @throws ExecutionException if an execution error occurs.
|
||||
* @throws ExecutionException if an execution error occurs.
|
||||
*/
|
||||
public static void main(String[] args) throws InterruptedException, ExecutionException {
|
||||
App app = new App();
|
||||
var app = new App();
|
||||
try {
|
||||
app.promiseUsage();
|
||||
} finally {
|
||||
@@ -100,13 +101,12 @@ public class App {
|
||||
* consume the result in a Consumer<Character>
|
||||
*/
|
||||
private void calculateLowestFrequencyChar() {
|
||||
lowestFrequencyChar()
|
||||
.thenAccept(
|
||||
charFrequency -> {
|
||||
LOGGER.info("Char with lowest frequency is: {}", charFrequency);
|
||||
taskCompleted();
|
||||
}
|
||||
);
|
||||
lowestFrequencyChar().thenAccept(
|
||||
charFrequency -> {
|
||||
LOGGER.info("Char with lowest frequency is: {}", charFrequency);
|
||||
taskCompleted();
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -114,13 +114,12 @@ public class App {
|
||||
* in a Consumer<Integer>
|
||||
*/
|
||||
private void calculateLineCount() {
|
||||
countLines()
|
||||
.thenAccept(
|
||||
count -> {
|
||||
LOGGER.info("Line count is: {}", count);
|
||||
taskCompleted();
|
||||
}
|
||||
);
|
||||
countLines().thenAccept(
|
||||
count -> {
|
||||
LOGGER.info("Line count is: {}", count);
|
||||
taskCompleted();
|
||||
}
|
||||
);
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -128,17 +127,15 @@ public class App {
|
||||
* then promise to apply function to calculate lowest character frequency.
|
||||
*/
|
||||
private Promise<Character> lowestFrequencyChar() {
|
||||
return characterFrequency()
|
||||
.thenApply(Utility::lowestFrequencyChar);
|
||||
return characterFrequency().thenApply(Utility::lowestFrequencyChar);
|
||||
}
|
||||
|
||||
/*
|
||||
* Download the file at DEFAULT_URL and when that promise is fulfilled,
|
||||
* then promise to apply function to calculate character frequency.
|
||||
*/
|
||||
private Promise<Map<Character, Integer>> characterFrequency() {
|
||||
return download(DEFAULT_URL)
|
||||
.thenApply(Utility::characterFrequency);
|
||||
private Promise<Map<Character, Long>> characterFrequency() {
|
||||
return download(DEFAULT_URL).thenApply(Utility::characterFrequency);
|
||||
}
|
||||
|
||||
/*
|
||||
@@ -146,8 +143,7 @@ public class App {
|
||||
* then promise to apply function to count lines in that file.
|
||||
*/
|
||||
private Promise<Integer> countLines() {
|
||||
return download(DEFAULT_URL)
|
||||
.thenApply(Utility::countLines);
|
||||
return download(DEFAULT_URL).thenApply(Utility::countLines);
|
||||
}
|
||||
|
||||
/*
|
||||
|
@@ -32,10 +32,10 @@ import java.util.function.Function;
|
||||
/**
|
||||
* A Promise represents a proxy for a value not necessarily known when the promise is created. It
|
||||
* allows you to associate dependent promises to an asynchronous action's eventual success value or
|
||||
* failure reason. This lets asynchronous methods return values like synchronous methods: instead
|
||||
* of the final value, the asynchronous method returns a promise of having a value at some point
|
||||
* in the future.
|
||||
*
|
||||
* failure reason. This lets asynchronous methods return values like synchronous methods: instead of
|
||||
* the final value, the asynchronous method returns a promise of having a value at some point in the
|
||||
* future.
|
||||
*
|
||||
* @param <T> type of result.
|
||||
*/
|
||||
public class Promise<T> extends PromiseSupport<T> {
|
||||
@@ -51,6 +51,7 @@ public class Promise<T> extends PromiseSupport<T> {
|
||||
|
||||
/**
|
||||
* Fulfills the promise with the provided value.
|
||||
*
|
||||
* @param value the fulfilled value that can be accessed using {@link #get()}.
|
||||
*/
|
||||
@Override
|
||||
@@ -61,8 +62,9 @@ public class Promise<T> extends PromiseSupport<T> {
|
||||
|
||||
/**
|
||||
* Fulfills the promise with exception due to error in execution.
|
||||
* @param exception the exception will be wrapped in {@link ExecutionException}
|
||||
* when accessing the value using {@link #get()}.
|
||||
*
|
||||
* @param exception the exception will be wrapped in {@link ExecutionException} when accessing the
|
||||
* value using {@link #get()}.
|
||||
*/
|
||||
@Override
|
||||
public void fulfillExceptionally(Exception exception) {
|
||||
@@ -86,10 +88,10 @@ public class Promise<T> extends PromiseSupport<T> {
|
||||
}
|
||||
|
||||
/**
|
||||
* Executes the task using the executor in other thread and fulfills the promise returned
|
||||
* once the task completes either successfully or with an exception.
|
||||
*
|
||||
* @param task the task that will provide the value to fulfill the promise.
|
||||
* Executes the task using the executor in other thread and fulfills the promise returned once the
|
||||
* task completes either successfully or with an exception.
|
||||
*
|
||||
* @param task the task that will provide the value to fulfill the promise.
|
||||
* @param executor the executor in which the task should be run.
|
||||
* @return a promise that represents the result of running the task provided.
|
||||
*/
|
||||
@@ -105,21 +107,23 @@ public class Promise<T> extends PromiseSupport<T> {
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a new promise that, when this promise is fulfilled normally, is fulfilled with
|
||||
* result of this promise as argument to the action provided.
|
||||
* Returns a new promise that, when this promise is fulfilled normally, is fulfilled with result
|
||||
* of this promise as argument to the action provided.
|
||||
*
|
||||
* @param action action to be executed.
|
||||
* @return a new promise.
|
||||
*/
|
||||
public Promise<Void> thenAccept(Consumer<? super T> action) {
|
||||
Promise<Void> dest = new Promise<>();
|
||||
var dest = new Promise<Void>();
|
||||
fulfillmentAction = new ConsumeAction(this, dest, action);
|
||||
return dest;
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Set the exception handler on this promise.
|
||||
* @param exceptionHandler a consumer that will handle the exception occurred while fulfilling
|
||||
* the promise.
|
||||
*
|
||||
* @param exceptionHandler a consumer that will handle the exception occurred while fulfilling the
|
||||
* promise.
|
||||
* @return this
|
||||
*/
|
||||
public Promise<T> onError(Consumer<? super Throwable> exceptionHandler) {
|
||||
@@ -128,8 +132,9 @@ public class Promise<T> extends PromiseSupport<T> {
|
||||
}
|
||||
|
||||
/**
|
||||
* Returns a new promise that, when this promise is fulfilled normally, is fulfilled with
|
||||
* result of this promise as argument to the function provided.
|
||||
* Returns a new promise that, when this promise is fulfilled normally, is fulfilled with result
|
||||
* of this promise as argument to the function provided.
|
||||
*
|
||||
* @param func function to be executed.
|
||||
* @return a new promise.
|
||||
*/
|
||||
@@ -140,8 +145,8 @@ public class Promise<T> extends PromiseSupport<T> {
|
||||
}
|
||||
|
||||
/**
|
||||
* Accesses the value from source promise and calls the consumer, then fulfills the
|
||||
* destination promise.
|
||||
* Accesses the value from source promise and calls the consumer, then fulfills the destination
|
||||
* promise.
|
||||
*/
|
||||
private class ConsumeAction implements Runnable {
|
||||
|
||||
|
@@ -27,16 +27,15 @@ import java.util.concurrent.ExecutionException;
|
||||
import java.util.concurrent.Future;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.TimeoutException;
|
||||
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
/**
|
||||
* A really simplified implementation of future that allows completing it successfully with a value
|
||||
* A really simplified implementation of future that allows completing it successfully with a value
|
||||
* or exceptionally with an exception.
|
||||
*/
|
||||
class PromiseSupport<T> implements Future<T> {
|
||||
|
||||
|
||||
private static final Logger LOGGER = LoggerFactory.getLogger(PromiseSupport.class);
|
||||
|
||||
private static final int RUNNING = 1;
|
||||
@@ -93,13 +92,12 @@ class PromiseSupport<T> implements Future<T> {
|
||||
}
|
||||
if (state == COMPLETED) {
|
||||
return value;
|
||||
}
|
||||
}
|
||||
throw new ExecutionException(exception);
|
||||
}
|
||||
|
||||
@Override
|
||||
public T get(long timeout, TimeUnit unit)
|
||||
throws ExecutionException, TimeoutException {
|
||||
public T get(long timeout, TimeUnit unit) throws ExecutionException {
|
||||
synchronized (lock) {
|
||||
while (state == RUNNING) {
|
||||
try {
|
||||
@@ -110,10 +108,10 @@ class PromiseSupport<T> implements Future<T> {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
if (state == COMPLETED) {
|
||||
return value;
|
||||
}
|
||||
}
|
||||
throw new ExecutionException(exception);
|
||||
}
|
||||
}
|
@@ -29,13 +29,13 @@ import java.io.FileReader;
|
||||
import java.io.FileWriter;
|
||||
import java.io.IOException;
|
||||
import java.io.InputStreamReader;
|
||||
import java.io.Reader;
|
||||
import java.net.URL;
|
||||
import java.util.HashMap;
|
||||
import java.util.Iterator;
|
||||
import java.util.Collections;
|
||||
import java.util.Comparator;
|
||||
import java.util.Map;
|
||||
import java.util.Map.Entry;
|
||||
|
||||
import java.util.function.Function;
|
||||
import java.util.stream.Collectors;
|
||||
import org.slf4j.Logger;
|
||||
import org.slf4j.LoggerFactory;
|
||||
|
||||
@@ -48,86 +48,68 @@ public class Utility {
|
||||
|
||||
/**
|
||||
* Calculates character frequency of the file provided.
|
||||
*
|
||||
* @param fileLocation location of the file.
|
||||
* @return a map of character to its frequency, an empty map if file does not exist.
|
||||
*/
|
||||
public static Map<Character, Integer> characterFrequency(String fileLocation) {
|
||||
Map<Character, Integer> characterToFrequency = new HashMap<>();
|
||||
try (Reader reader = new FileReader(fileLocation);
|
||||
BufferedReader bufferedReader = new BufferedReader(reader)) {
|
||||
for (String line; (line = bufferedReader.readLine()) != null;) {
|
||||
for (char c : line.toCharArray()) {
|
||||
if (!characterToFrequency.containsKey(c)) {
|
||||
characterToFrequency.put(c, 1);
|
||||
} else {
|
||||
characterToFrequency.put(c, characterToFrequency.get(c) + 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
public static Map<Character, Long> characterFrequency(String fileLocation) {
|
||||
try (var bufferedReader = new BufferedReader(new FileReader(fileLocation))) {
|
||||
return bufferedReader.lines()
|
||||
.flatMapToInt(String::chars)
|
||||
.mapToObj(x -> (char) x)
|
||||
.collect(Collectors.groupingBy(Function.identity(), Collectors.counting()));
|
||||
} catch (IOException ex) {
|
||||
ex.printStackTrace();
|
||||
}
|
||||
return characterToFrequency;
|
||||
return Collections.emptyMap();
|
||||
}
|
||||
|
||||
/**
|
||||
* Return the character with the lowest frequency, if exists.
|
||||
*
|
||||
* @return the character, {@code Optional.empty()} otherwise.
|
||||
*/
|
||||
public static Character lowestFrequencyChar(Map<Character, Integer> charFrequency) {
|
||||
Character lowestFrequencyChar = null;
|
||||
Iterator<Entry<Character, Integer>> iterator = charFrequency.entrySet().iterator();
|
||||
Entry<Character, Integer> entry = iterator.next();
|
||||
int minFrequency = entry.getValue();
|
||||
lowestFrequencyChar = entry.getKey();
|
||||
|
||||
while (iterator.hasNext()) {
|
||||
entry = iterator.next();
|
||||
if (entry.getValue() < minFrequency) {
|
||||
minFrequency = entry.getValue();
|
||||
lowestFrequencyChar = entry.getKey();
|
||||
}
|
||||
}
|
||||
|
||||
return lowestFrequencyChar;
|
||||
public static Character lowestFrequencyChar(Map<Character, Long> charFrequency) {
|
||||
return charFrequency
|
||||
.entrySet()
|
||||
.stream()
|
||||
.min(Comparator.comparingLong(Entry::getValue))
|
||||
.map(Entry::getKey)
|
||||
.orElseThrow();
|
||||
}
|
||||
|
||||
/**
|
||||
* Count the number of lines in a file.
|
||||
*
|
||||
* @return number of lines, 0 if file does not exist.
|
||||
*/
|
||||
public static Integer countLines(String fileLocation) {
|
||||
int lineCount = 0;
|
||||
try (Reader reader = new FileReader(fileLocation);
|
||||
BufferedReader bufferedReader = new BufferedReader(reader)) {
|
||||
while (bufferedReader.readLine() != null) {
|
||||
lineCount++;
|
||||
}
|
||||
try (var bufferedReader = new BufferedReader(new FileReader(fileLocation))) {
|
||||
return (int) bufferedReader.lines().count();
|
||||
} catch (IOException ex) {
|
||||
ex.printStackTrace();
|
||||
}
|
||||
return lineCount;
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* Downloads the contents from the given urlString, and stores it in a temporary directory.
|
||||
*
|
||||
* @return the absolute path of the file downloaded.
|
||||
*/
|
||||
public static String downloadFile(String urlString) throws IOException {
|
||||
LOGGER.info("Downloading contents from url: {}", urlString);
|
||||
URL url = new URL(urlString);
|
||||
File file = File.createTempFile("promise_pattern", null);
|
||||
try (Reader reader = new InputStreamReader(url.openStream());
|
||||
BufferedReader bufferedReader = new BufferedReader(reader);
|
||||
FileWriter writer = new FileWriter(file)) {
|
||||
for (String line; (line = bufferedReader.readLine()) != null; ) {
|
||||
var url = new URL(urlString);
|
||||
var file = File.createTempFile("promise_pattern", null);
|
||||
try (var bufferedReader = new BufferedReader(new InputStreamReader(url.openStream()));
|
||||
var writer = new FileWriter(file)) {
|
||||
String line;
|
||||
while ((line = bufferedReader.readLine()) != null) {
|
||||
writer.write(line);
|
||||
writer.write("\n");
|
||||
}
|
||||
LOGGER.info("File downloaded at: {}", file.getAbsolutePath());
|
||||
return file.getAbsolutePath();
|
||||
} catch (IOException ex) {
|
||||
throw ex;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
@@ -23,12 +23,10 @@
|
||||
|
||||
package com.iluwatar.promise;
|
||||
|
||||
import java.util.concurrent.ExecutionException;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
import java.util.concurrent.ExecutionException;
|
||||
|
||||
/**
|
||||
*
|
||||
* Application test.
|
||||
*/
|
||||
public class AppTest {
|
||||
|
@@ -23,8 +23,13 @@
|
||||
|
||||
package com.iluwatar.promise;
|
||||
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
import static org.junit.jupiter.api.Assertions.assertFalse;
|
||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||
import static org.junit.jupiter.api.Assertions.fail;
|
||||
import static org.mockito.Matchers.eq;
|
||||
import static org.mockito.Mockito.mock;
|
||||
import static org.mockito.Mockito.verify;
|
||||
|
||||
import java.util.concurrent.Callable;
|
||||
import java.util.concurrent.ExecutionException;
|
||||
@@ -33,15 +38,8 @@ import java.util.concurrent.Executors;
|
||||
import java.util.concurrent.TimeUnit;
|
||||
import java.util.concurrent.TimeoutException;
|
||||
import java.util.function.Consumer;
|
||||
import java.util.function.Function;
|
||||
|
||||
import static org.junit.jupiter.api.Assertions.assertEquals;
|
||||
import static org.junit.jupiter.api.Assertions.assertFalse;
|
||||
import static org.junit.jupiter.api.Assertions.assertTrue;
|
||||
import static org.junit.jupiter.api.Assertions.fail;
|
||||
import static org.mockito.Matchers.eq;
|
||||
import static org.mockito.Mockito.mock;
|
||||
import static org.mockito.Mockito.verify;
|
||||
import org.junit.jupiter.api.BeforeEach;
|
||||
import org.junit.jupiter.api.Test;
|
||||
|
||||
/**
|
||||
* Tests Promise class.
|
||||
@@ -58,7 +56,7 @@ public class PromiseTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void promiseIsFulfilledWithTheResultantValueOfExecutingTheTask()
|
||||
public void promiseIsFulfilledWithTheResultantValueOfExecutingTheTask()
|
||||
throws InterruptedException, ExecutionException {
|
||||
promise.fulfillInAsync(new NumberCrunchingTask(), executor);
|
||||
|
||||
@@ -66,21 +64,20 @@ public class PromiseTest {
|
||||
assertTrue(promise.isDone());
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
public void promiseIsFulfilledWithAnExceptionIfTaskThrowsAnException()
|
||||
throws InterruptedException, ExecutionException, TimeoutException {
|
||||
public void promiseIsFulfilledWithAnExceptionIfTaskThrowsAnException()
|
||||
throws InterruptedException, TimeoutException {
|
||||
testWaitingForeverForPromiseToBeFulfilled();
|
||||
testWaitingSomeTimeForPromiseToBeFulfilled();
|
||||
}
|
||||
|
||||
private void testWaitingForeverForPromiseToBeFulfilled()
|
||||
throws InterruptedException, TimeoutException {
|
||||
Promise<Integer> promise = new Promise<>();
|
||||
private void testWaitingForeverForPromiseToBeFulfilled() throws InterruptedException {
|
||||
var promise = new Promise<Integer>();
|
||||
promise.fulfillInAsync(() -> {
|
||||
throw new RuntimeException("Barf!");
|
||||
}, executor);
|
||||
|
||||
|
||||
try {
|
||||
promise.get();
|
||||
fail("Fetching promise should result in exception if the task threw an exception");
|
||||
@@ -88,7 +85,7 @@ public class PromiseTest {
|
||||
assertTrue(promise.isDone());
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
promise.get(1000, TimeUnit.SECONDS);
|
||||
fail("Fetching promise should result in exception if the task threw an exception");
|
||||
@@ -97,14 +94,13 @@ public class PromiseTest {
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
}
|
||||
|
||||
private void testWaitingSomeTimeForPromiseToBeFulfilled()
|
||||
throws InterruptedException, TimeoutException {
|
||||
Promise<Integer> promise = new Promise<>();
|
||||
|
||||
private void testWaitingSomeTimeForPromiseToBeFulfilled() throws InterruptedException {
|
||||
var promise = new Promise<Integer>();
|
||||
promise.fulfillInAsync(() -> {
|
||||
throw new RuntimeException("Barf!");
|
||||
}, executor);
|
||||
|
||||
|
||||
try {
|
||||
promise.get(1000, TimeUnit.SECONDS);
|
||||
fail("Fetching promise should result in exception if the task threw an exception");
|
||||
@@ -112,7 +108,7 @@ public class PromiseTest {
|
||||
assertTrue(promise.isDone());
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
promise.get();
|
||||
fail("Fetching promise should result in exception if the task threw an exception");
|
||||
@@ -120,18 +116,15 @@ public class PromiseTest {
|
||||
assertTrue(promise.isDone());
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
}
|
||||
|
||||
@Test
|
||||
public void dependentPromiseIsFulfilledAfterTheConsumerConsumesTheResultOfThisPromise()
|
||||
public void dependentPromiseIsFulfilledAfterTheConsumerConsumesTheResultOfThisPromise()
|
||||
throws InterruptedException, ExecutionException {
|
||||
Promise<Void> dependentPromise = promise
|
||||
var dependentPromise = promise
|
||||
.fulfillInAsync(new NumberCrunchingTask(), executor)
|
||||
.thenAccept(value -> {
|
||||
assertEquals(NumberCrunchingTask.CRUNCHED_NUMBER, value);
|
||||
});
|
||||
|
||||
.thenAccept(value -> assertEquals(NumberCrunchingTask.CRUNCHED_NUMBER, value));
|
||||
|
||||
dependentPromise.get();
|
||||
assertTrue(dependentPromise.isDone());
|
||||
@@ -139,9 +132,9 @@ public class PromiseTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void dependentPromiseIsFulfilledWithAnExceptionIfConsumerThrowsAnException()
|
||||
throws InterruptedException, ExecutionException, TimeoutException {
|
||||
Promise<Void> dependentPromise = promise
|
||||
public void dependentPromiseIsFulfilledWithAnExceptionIfConsumerThrowsAnException()
|
||||
throws InterruptedException {
|
||||
var dependentPromise = promise
|
||||
.fulfillInAsync(new NumberCrunchingTask(), executor)
|
||||
.thenAccept(value -> {
|
||||
throw new RuntimeException("Barf!");
|
||||
@@ -155,7 +148,7 @@ public class PromiseTest {
|
||||
assertTrue(promise.isDone());
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
dependentPromise.get(1000, TimeUnit.SECONDS);
|
||||
fail("Fetching dependent promise should result in exception "
|
||||
@@ -167,12 +160,12 @@ public class PromiseTest {
|
||||
}
|
||||
|
||||
@Test
|
||||
public void dependentPromiseIsFulfilledAfterTheFunctionTransformsTheResultOfThisPromise()
|
||||
public void dependentPromiseIsFulfilledAfterTheFunctionTransformsTheResultOfThisPromise()
|
||||
throws InterruptedException, ExecutionException {
|
||||
Promise<String> dependentPromise = promise
|
||||
var dependentPromise = promise
|
||||
.fulfillInAsync(new NumberCrunchingTask(), executor)
|
||||
.thenApply(value -> {
|
||||
assertEquals(NumberCrunchingTask.CRUNCHED_NUMBER, value);
|
||||
assertEquals(NumberCrunchingTask.CRUNCHED_NUMBER, value);
|
||||
return String.valueOf(value);
|
||||
});
|
||||
|
||||
@@ -181,11 +174,11 @@ public class PromiseTest {
|
||||
assertTrue(dependentPromise.isDone());
|
||||
assertFalse(dependentPromise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
public void dependentPromiseIsFulfilledWithAnExceptionIfTheFunctionThrowsException()
|
||||
throws InterruptedException, ExecutionException, TimeoutException {
|
||||
Promise<String> dependentPromise = promise
|
||||
public void dependentPromiseIsFulfilledWithAnExceptionIfTheFunctionThrowsException()
|
||||
throws InterruptedException {
|
||||
var dependentPromise = promise
|
||||
.fulfillInAsync(new NumberCrunchingTask(), executor)
|
||||
.thenApply(value -> {
|
||||
throw new RuntimeException("Barf!");
|
||||
@@ -199,7 +192,7 @@ public class PromiseTest {
|
||||
assertTrue(promise.isDone());
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
|
||||
|
||||
try {
|
||||
dependentPromise.get(1000, TimeUnit.SECONDS);
|
||||
fail("Fetching dependent promise should result in exception "
|
||||
@@ -209,26 +202,26 @@ public class PromiseTest {
|
||||
assertFalse(promise.isCancelled());
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@Test
|
||||
public void fetchingAnAlreadyFulfilledPromiseReturnsTheFulfilledValueImmediately()
|
||||
throws InterruptedException, ExecutionException, TimeoutException {
|
||||
Promise<Integer> promise = new Promise<>();
|
||||
public void fetchingAnAlreadyFulfilledPromiseReturnsTheFulfilledValueImmediately()
|
||||
throws ExecutionException {
|
||||
var promise = new Promise<Integer>();
|
||||
promise.fulfill(NumberCrunchingTask.CRUNCHED_NUMBER);
|
||||
|
||||
|
||||
promise.get(1000, TimeUnit.SECONDS);
|
||||
}
|
||||
|
||||
|
||||
@SuppressWarnings("unchecked")
|
||||
@Test
|
||||
public void exceptionHandlerIsCalledWhenPromiseIsFulfilledExceptionally() {
|
||||
Promise<Object> promise = new Promise<>();
|
||||
Consumer<Throwable> exceptionHandler = mock(Consumer.class);
|
||||
var promise = new Promise<>();
|
||||
var exceptionHandler = mock(Consumer.class);
|
||||
promise.onError(exceptionHandler);
|
||||
|
||||
Exception exception = new Exception("barf!");
|
||||
|
||||
var exception = new Exception("barf!");
|
||||
promise.fulfillExceptionally(exception);
|
||||
|
||||
|
||||
verify(exceptionHandler).accept(eq(exception));
|
||||
}
|
||||
|
||||
|
Reference in New Issue
Block a user