AsynchronousBodyInputStreamTest.java
/*
* Copyright (c) 2015, 2022 Oracle and/or its affiliates. All rights reserved.
*
* This program and the accompanying materials are made available under the
* terms of the Eclipse Public License v. 2.0, which is available at
* http://www.eclipse.org/legal/epl-2.0.
*
* This Source Code may also be made available under the following Secondary
* Licenses when the conditions for such availability set forth in the
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
* version 2 with the GNU Classpath Exception, which is available at
* https://www.gnu.org/software/classpath/license.html.
*
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
*/
package org.glassfish.jersey.jdk.connector.internal;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.util.concurrent.Callable;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicReference;
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.assertNotEquals;
import static org.junit.jupiter.api.Assertions.assertNull;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.fail;
/**
* @author Petr Janouch
*/
public class AsynchronousBodyInputStreamTest {
@Test
public void testBasicAsyncRead() {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
doTestBasicAsyncRead(stream, new TestReadListener(stream));
}
@Test
public void testBasicAsyncArrayRead() {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
doTestBasicAsyncRead(stream, new TestReadListener(stream, 15));
}
@Test
public void testBasicAsyncReadWithException() {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
doTestBasicAsyncReadWithException(stream, new TestReadListener(stream));
}
@Test
public void testBasicAsyncArrayReadWithException() {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
doTestBasicAsyncReadWithException(stream, new TestReadListener(stream, 15));
}
@Test
public void testListenerExecutor() throws InterruptedException {
final AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
ExecutorService executor = Executors.newSingleThreadExecutor();
Thread mainThread = Thread.currentThread();
final AtomicReference<Thread> dataAvailableThread = new AtomicReference<>();
final AtomicReference<Thread> allDataReadThread = new AtomicReference<>();
final CountDownLatch latch = new CountDownLatch(1);
try {
stream.setListenerExecutor(executor);
stream.setReadListener(new ReadListener() {
@Override
public void onDataAvailable() throws IOException {
dataAvailableThread.set(Thread.currentThread());
while (stream.isReady()) {
stream.read();
}
}
@Override
public void onAllDataRead() {
allDataReadThread.set(Thread.currentThread());
latch.countDown();
}
@Override
public void onError(Throwable t) {
}
});
stream.notifyDataAvailable(stringToBuffer("Message"));
stream.notifyAllDataRead();
assertTrue(latch.await(5, TimeUnit.SECONDS));
} finally {
executor.shutdownNow();
}
assertNotEquals(mainThread, dataAvailableThread.get());
assertNotEquals(mainThread, allDataReadThread.get());
}
@Test
public void testDataBeforeAsyncModeCommit() {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
String msg3 = "CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC";
String msg4 = "DDDDD";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
TestReadListener readListener = new TestReadListener(stream);
stream.setReadListener(readListener);
assertEquals(msg1 + msg2, readListener.getReceivedData());
stream.notifyDataAvailable(stringToBuffer(msg3));
stream.notifyDataAvailable(stringToBuffer(msg4));
stream.notifyAllDataRead();
assertEquals(msg1 + msg2 + msg3 + msg4, readListener.getReceivedData());
assertTrue(readListener.isAllDataRead());
assertNull(readListener.getError());
}
@Test
public void testDataBeforeSyncModeCommit() throws IOException {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
String msg3 = "CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC";
String msg4 = "DDDDD";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
assertEquals((msg1 + msg2).length(), stream.available());
stream.notifyDataAvailable(stringToBuffer(msg3));
stream.notifyDataAvailable(stringToBuffer(msg4));
assertEquals((msg1 + msg2 + msg3 + msg4).length(), stream.available());
}
@Test
public void testAllDataBeforeAsyncModeCommit() {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
stream.notifyAllDataRead();
TestReadListener readListener = new TestReadListener(stream);
stream.setReadListener(readListener);
assertEquals(msg1 + msg2, readListener.getReceivedData());
assertTrue(readListener.isAllDataRead());
assertNull(readListener.getError());
}
@Test
public void testAllDataBeforeSyncModeCommit() throws IOException {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
stream.notifyAllDataRead();
assertEquals((msg1 + msg2).length(), stream.available());
}
@Test
public void testErrorBeforeAsyncModeCommit() throws IOException {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
Throwable t = new Throwable();
stream.notifyError(t);
TestReadListener readListener = new TestReadListener(stream);
stream.setReadListener(readListener);
assertEquals(msg1 + msg2, readListener.getReceivedData());
assertFalse(readListener.isAllDataRead());
assertTrue(readListener.getError() == t);
}
@Test
public void testErrorBeforeSyncModeCommit() throws IOException {
AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
Throwable t = new Throwable();
stream.notifyError(t);
try {
stream.available();
fail();
} catch (Throwable e) {
assertTrue(e.getCause() == t);
}
}
@Test
public void testUnsupportedSync() {
final AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
try {
// touch this stream to make it synchronous
stream.tryRead();
} catch (IOException e) {
e.printStackTrace();
fail();
}
assertUnsupported(() -> {
stream.isReady();
return null;
});
assertUnsupported(() -> {
stream.setReadListener(new TestReadListener(stream));
return null;
});
assertUnsupported(() -> {
ExecutorService executorService = Executors.newSingleThreadExecutor();
try {
stream.setListenerExecutor(executorService);
} finally {
executorService.shutdownNow();
}
return null;
});
}
@Test
public void testUnsupportedAsync() {
final AsynchronousBodyInputStream stream = new AsynchronousBodyInputStream();
stream.setReadListener(new TestReadListener(stream));
assertUnsupported(() -> {
stream.tryRead();
return null;
});
assertUnsupported(() -> {
stream.tryRead(new byte[10]);
return null;
});
assertUnsupported(() -> {
stream.tryRead(new byte[10], 0, 10);
return null;
});
assertUnsupported(() -> stream.skip(10));
assertUnsupported(() -> stream.available());
}
private void doTestBasicAsyncRead(AsynchronousBodyInputStream stream, TestReadListener readListener) {
stream.setReadListener(readListener);
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
String msg3 = "CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
stream.notifyDataAvailable(stringToBuffer(msg3));
stream.notifyAllDataRead();
assertEquals(msg1 + msg2 + msg3, readListener.getReceivedData());
assertTrue(readListener.isAllDataRead());
assertNull(readListener.getError());
}
private void doTestBasicAsyncReadWithException(AsynchronousBodyInputStream stream, TestReadListener readListener) {
stream.setReadListener(readListener);
String msg1 = "AAAAAAAAAAAAAAAAAAAA";
String msg2 = "BBBBBBBBBBBBB";
String msg3 = "CCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCCC";
stream.notifyDataAvailable(stringToBuffer(msg1));
stream.notifyDataAvailable(stringToBuffer(msg2));
stream.notifyDataAvailable(stringToBuffer(msg3));
Throwable t = new Throwable();
stream.notifyError(t);
assertEquals(msg1 + msg2 + msg3, readListener.getReceivedData());
assertFalse(readListener.isAllDataRead());
assertTrue(readListener.getError() == t);
}
private static void assertUnsupported(Callable unsupported) {
try {
unsupported.call();
fail();
} catch (UnsupportedOperationException e) {
// expected
} catch (Exception e) {
e.printStackTrace();
fail();
}
}
private static ByteBuffer stringToBuffer(String str) {
return ByteBuffer.wrap(str.getBytes());
}
private static class TestReadListener implements ReadListener {
private final ByteArrayOutputStream receivedData = new ByteArrayOutputStream();
private final AsynchronousBodyInputStream inputStream;
private final int inputArraySize;
private volatile Throwable error = null;
private volatile boolean allDataRead = false;
private volatile boolean listenerCallExpected = true;
public TestReadListener(AsynchronousBodyInputStream inputStream, int inputArraySize) {
this.inputStream = inputStream;
this.inputArraySize = inputArraySize;
}
public TestReadListener(AsynchronousBodyInputStream inputStream) {
this.inputStream = inputStream;
this.inputArraySize = -1;
}
@Override
public void onDataAvailable() throws IOException {
if (!listenerCallExpected) {
fail();
}
listenerCallExpected = false;
while (inputStream.isReady()) {
if (inputArraySize == -1) {
receivedData.write(inputStream.read());
} else {
byte[] inputArray = new byte[inputArraySize];
int read = inputStream.read(inputArray);
receivedData.write(inputArray, 0, read);
}
}
listenerCallExpected = true;
}
@Override
public void onAllDataRead() {
allDataRead = true;
}
@Override
public void onError(Throwable t) {
error = t;
}
public boolean isAllDataRead() {
return allDataRead;
}
public Throwable getError() {
return error;
}
public String getReceivedData() {
return new String(receivedData.toByteArray());
}
}
}