| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You under the Apache License, Version 2.0 |
| * (the "License"); you may not use this file except in compliance with |
| * the License. You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| package org.apache.commons.io.input; |
| |
| import static org.junit.jupiter.api.Assertions.assertEquals; |
| import static org.junit.jupiter.api.Assertions.assertThrows; |
| import static org.junit.jupiter.api.Assertions.assertTimeout; |
| import static org.junit.jupiter.api.Assertions.assertTrue; |
| |
| import java.io.BufferedInputStream; |
| import java.io.BufferedOutputStream; |
| import java.io.ByteArrayOutputStream; |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.nio.charset.StandardCharsets; |
| import java.time.Duration; |
| import java.util.concurrent.BlockingQueue; |
| import java.util.concurrent.CountDownLatch; |
| import java.util.concurrent.LinkedBlockingQueue; |
| import java.util.concurrent.TimeUnit; |
| import java.util.concurrent.atomic.AtomicBoolean; |
| import java.util.stream.Stream; |
| |
| import org.apache.commons.io.IOUtils; |
| import org.apache.commons.io.output.QueueOutputStream; |
| import org.apache.commons.io.output.QueueOutputStreamTest; |
| import org.apache.commons.lang3.StringUtils; |
| import org.junit.jupiter.api.DisplayName; |
| import org.junit.jupiter.api.Test; |
| import org.junit.jupiter.params.ParameterizedTest; |
| import org.junit.jupiter.params.provider.Arguments; |
| import org.junit.jupiter.params.provider.MethodSource; |
| |
| import com.google.common.base.Stopwatch; |
| |
| /** |
| * Test {@link QueueInputStream}. |
| * |
| * @see QueueOutputStreamTest |
| */ |
| public class QueueInputStreamTest { |
| |
| public static Stream<Arguments> inputData() { |
| // @formatter:off |
| return Stream.of(Arguments.of(""), |
| Arguments.of("1"), |
| Arguments.of("12"), |
| Arguments.of("1234"), |
| Arguments.of("12345678"), |
| Arguments.of(StringUtils.repeat("A", 4095)), |
| Arguments.of(StringUtils.repeat("A", 4096)), |
| Arguments.of(StringUtils.repeat("A", 4097)), |
| Arguments.of(StringUtils.repeat("A", 8191)), |
| Arguments.of(StringUtils.repeat("A", 8192)), |
| Arguments.of(StringUtils.repeat("A", 8193)), |
| Arguments.of(StringUtils.repeat("A", 8192 * 4))); |
| // @formatter:on |
| } |
| |
| private int defaultBufferSize() { |
| return 8192; |
| } |
| |
| private String readUnbuffered(final InputStream inputStream) throws IOException { |
| return readUnbuffered(inputStream, Integer.MAX_VALUE); |
| } |
| |
| private String readUnbuffered(final InputStream inputStream, final int maxBytes) throws IOException { |
| if (maxBytes == 0) { |
| return ""; |
| } |
| |
| final ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream(); |
| int n = -1; |
| while ((n = inputStream.read()) != -1) { |
| byteArrayOutputStream.write(n); |
| if (byteArrayOutputStream.size() >= maxBytes) { |
| break; |
| } |
| } |
| return byteArrayOutputStream.toString(StandardCharsets.UTF_8.name()); |
| } |
| |
| @ParameterizedTest(name = "inputData={0}") |
| @MethodSource("inputData") |
| public void testBufferedReads(final String inputData) throws IOException { |
| final BlockingQueue<Integer> queue = new LinkedBlockingQueue<>(); |
| try (BufferedInputStream inputStream = new BufferedInputStream(new QueueInputStream(queue)); |
| final QueueOutputStream outputStream = new QueueOutputStream(queue)) { |
| outputStream.write(inputData.getBytes(StandardCharsets.UTF_8)); |
| final String actualData = IOUtils.toString(inputStream, StandardCharsets.UTF_8); |
| assertEquals(inputData, actualData); |
| } |
| } |
| |
| @ParameterizedTest(name = "inputData={0}") |
| @MethodSource("inputData") |
| public void testBufferedReadWrite(final String inputData) throws IOException { |
| final BlockingQueue<Integer> queue = new LinkedBlockingQueue<>(); |
| try (BufferedInputStream inputStream = new BufferedInputStream(new QueueInputStream(queue)); |
| final BufferedOutputStream outputStream = new BufferedOutputStream(new QueueOutputStream(queue), defaultBufferSize())) { |
| outputStream.write(inputData.getBytes(StandardCharsets.UTF_8)); |
| outputStream.flush(); |
| final String dataCopy = IOUtils.toString(inputStream, StandardCharsets.UTF_8); |
| assertEquals(inputData, dataCopy); |
| } |
| } |
| |
| @ParameterizedTest(name = "inputData={0}") |
| @MethodSource("inputData") |
| public void testBufferedWrites(final String inputData) throws IOException { |
| final BlockingQueue<Integer> queue = new LinkedBlockingQueue<>(); |
| try (QueueInputStream inputStream = new QueueInputStream(queue); |
| final BufferedOutputStream outputStream = new BufferedOutputStream(new QueueOutputStream(queue), defaultBufferSize())) { |
| outputStream.write(inputData.getBytes(StandardCharsets.UTF_8)); |
| outputStream.flush(); |
| final String actualData = readUnbuffered(inputStream); |
| assertEquals(inputData, actualData); |
| } |
| } |
| |
| @Test |
| public void testInvalidArguments() { |
| assertThrows(NullPointerException.class, () -> new QueueInputStream(null), "queue is required"); |
| assertThrows(IllegalArgumentException.class, () -> QueueInputStream.builder().setTimeout(Duration.ofMillis(-1)).get(), "waitTime must not be negative"); |
| } |
| |
| @Test |
| public void testResetArguments() throws IOException { |
| try (QueueInputStream queueInputStream = QueueInputStream.builder().setTimeout(null).get()) { |
| assertEquals(Duration.ZERO, queueInputStream.getTimeout()); |
| assertEquals(0, queueInputStream.getBlockingQueue().size()); |
| } |
| try (QueueInputStream queueInputStream = QueueInputStream.builder().setBlockingQueue(null).get()) { |
| assertEquals(Duration.ZERO, queueInputStream.getTimeout()); |
| assertEquals(0, queueInputStream.getBlockingQueue().size()); |
| } |
| } |
| |
| @Test |
| @DisplayName("If read is interrupted while waiting, then exception is thrown") |
| public void testTimeoutInterrupted() throws Exception { |
| try (QueueInputStream inputStream = QueueInputStream.builder().setTimeout(Duration.ofMinutes(2)).get(); |
| final QueueOutputStream outputStream = inputStream.newQueueOutputStream()) { |
| |
| // read in a background thread |
| final AtomicBoolean result = new AtomicBoolean(); |
| final CountDownLatch latch = new CountDownLatch(1); |
| final Thread thread = new Thread(() -> { |
| // when thread is interrupted, verify ... |
| assertThrows(IllegalStateException.class, () -> readUnbuffered(inputStream, 3)); |
| assertTrue(Thread.currentThread().isInterrupted()); |
| result.set(true); |
| latch.countDown(); |
| }); |
| thread.setDaemon(true); |
| thread.start(); |
| |
| // interrupt and check that verification completed |
| thread.interrupt(); |
| latch.await(500, TimeUnit.MILLISECONDS); |
| assertTrue(result.get()); |
| } |
| } |
| |
| @Test |
| @DisplayName("If data is not available in queue, then read will wait until wait time elapses") |
| public void testTimeoutUnavailableData() throws IOException { |
| try (QueueInputStream inputStream = QueueInputStream.builder().setTimeout(Duration.ofMillis(500)).get(); |
| final QueueOutputStream outputStream = inputStream.newQueueOutputStream()) { |
| final Stopwatch stopwatch = Stopwatch.createStarted(); |
| final String actualData = assertTimeout(Duration.ofSeconds(1), () -> readUnbuffered(inputStream, 3)); |
| stopwatch.stop(); |
| assertEquals("", actualData); |
| |
| assertTrue(stopwatch.elapsed(TimeUnit.MILLISECONDS) >= 500, () -> stopwatch.toString()); |
| } |
| } |
| |
| @ParameterizedTest(name = "inputData={0}") |
| @MethodSource("inputData") |
| public void testUnbufferedReadWrite(final String inputData) throws IOException { |
| try (QueueInputStream inputStream = new QueueInputStream(); |
| final QueueOutputStream outputStream = inputStream.newQueueOutputStream()) { |
| writeUnbuffered(outputStream, inputData); |
| final String actualData = readUnbuffered(inputStream); |
| assertEquals(inputData, actualData); |
| } |
| } |
| |
| @ParameterizedTest(name = "inputData={0}") |
| @MethodSource("inputData") |
| public void testUnbufferedReadWriteWithTimeout(final String inputData) throws IOException { |
| final Duration timeout = Duration.ofMinutes(2); |
| try (QueueInputStream inputStream = QueueInputStream.builder().setTimeout(timeout).get(); |
| final QueueOutputStream outputStream = inputStream.newQueueOutputStream()) { |
| assertEquals(timeout, inputStream.getTimeout()); |
| writeUnbuffered(outputStream, inputData); |
| final String actualData = assertTimeout(Duration.ofSeconds(1), () -> readUnbuffered(inputStream, inputData.length())); |
| assertEquals(inputData, actualData); |
| } |
| } |
| |
| private void writeUnbuffered(final QueueOutputStream outputStream, final String inputData) throws IOException { |
| final byte[] bytes = inputData.getBytes(StandardCharsets.UTF_8); |
| outputStream.write(bytes, 0, bytes.length); |
| } |
| } |