CommittingOutputStreamTest.java
/*
* Copyright (c) 2013, 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.tests.e2e.common.message.internal;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.HashMap;
import java.util.Map;
import javax.ws.rs.RuntimeType;
import javax.ws.rs.core.Configuration;
import org.glassfish.jersey.CommonProperties;
import org.glassfish.jersey.internal.util.PropertiesHelper;
import org.glassfish.jersey.message.internal.CommittingOutputStream;
import org.glassfish.jersey.message.internal.OutboundMessageContext;
import org.glassfish.jersey.model.internal.CommonConfig;
import org.glassfish.jersey.model.internal.ComponentBag;
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;
/**
* Test the {@link CommittingOutputStream}.
*
* @author Miroslav Fuksa
*/
public class CommittingOutputStreamTest {
private static class Passed {
boolean b;
public void pass() {
b = true;
}
}
@Test
public void testExactSizeOfBuffer() throws IOException {
final Passed passed = new Passed();
final ByteArrayOutputStream baos = new ByteArrayOutputStream(1000);
CommittingOutputStream cos = new CommittingOutputStream();
setupBufferedStreamProvider(passed, baos, cos, 3);
cos.write((byte) 1);
cos.write((byte) 2);
cos.write((byte) 3);
checkNotYetCommitted(passed, baos, cos);
cos.commit();
check(baos, new byte[]{1, 2, 3});
assertTrue(passed.b);
cos.close();
}
private void checkNotYetCommitted(Passed passed, ByteArrayOutputStream baos, CommittingOutputStream cos) {
assertFalse(passed.b);
assertFalse(cos.isCommitted());
check(baos, null);
}
private void checkCommitted(Passed passed, CommittingOutputStream cos) {
assertTrue(passed.b);
assertTrue(cos.isCommitted());
}
private void setupBufferedStreamProvider(final Passed passed, final ByteArrayOutputStream baos, CommittingOutputStream cos,
final int expectedContentLength) {
cos.setStreamProvider(new OutboundMessageContext.StreamProvider() {
@Override
public OutputStream getOutputStream(int contentLength) throws IOException {
assertEquals(expectedContentLength, contentLength);
passed.pass();
return baos;
}
});
cos.enableBuffering(3);
}
private void setupStreamProvider(final Passed passed, final ByteArrayOutputStream baos, CommittingOutputStream cos) {
cos.setStreamProvider(new OutboundMessageContext.StreamProvider() {
@Override
public OutputStream getOutputStream(int contentLength) throws IOException {
passed.pass();
return baos;
}
});
}
@Test
public void testExactSizeOfBufferByClose() throws IOException {
final Passed passed = new Passed();
final ByteArrayOutputStream baos = new ByteArrayOutputStream(1000);
CommittingOutputStream cos = new CommittingOutputStream();
setupBufferedStreamProvider(passed, baos, cos, 3);
cos.write((byte) 1);
cos.write((byte) 2);
cos.write((byte) 3);
checkNotYetCommitted(passed, baos, cos);
cos.close();
check(baos, new byte[]{1, 2, 3});
assertTrue(passed.b);
}
@Test
public void testLessBytesThanLimit() throws IOException {
final Passed passed = new Passed();
final ByteArrayOutputStream baos = new ByteArrayOutputStream(1000);
CommittingOutputStream cos = new CommittingOutputStream();
setupBufferedStreamProvider(passed, baos, cos, 2);
cos.write((byte) 1);
cos.write((byte) 2);
checkNotYetCommitted(passed, baos, cos);
cos.commit();
check(baos, new byte[]{1, 2});
cos.close();
assertTrue(passed.b);
}
@Test
public void testNoBytes() throws IOException {
final Passed passed = new Passed();
final ByteArrayOutputStream baos = new ByteArrayOutputStream(1000);
CommittingOutputStream cos = new CommittingOutputStream();
setupBufferedStreamProvider(passed, baos, cos, 0);
checkNotYetCommitted(passed, baos, cos);
cos.commit();
check(baos, null);
cos.close();
assertTrue(passed.b);
}
@Test
public void testBufferOverflow() throws IOException {
final Passed passed = new Passed();
final ByteArrayOutputStream baos = new ByteArrayOutputStream(1000);
CommittingOutputStream cos = new CommittingOutputStream();
setupBufferedStreamProvider(passed, baos, cos, -1);
cos.write((byte) 1);
cos.write((byte) 2);
cos.write((byte) 3);
checkNotYetCommitted(passed, baos, cos);
cos.write((byte) 4);
check(baos, new byte[]{1, 2, 3, 4});
cos.write((byte) 5);
check(baos, new byte[]{1, 2, 3, 4, 5});
cos.commit();
check(baos, new byte[]{1, 2, 3, 4, 5});
cos.close();
}
@Test
public void testNotBufferedOS() throws IOException {
final Passed passed = new Passed();
final ByteArrayOutputStream baos = new ByteArrayOutputStream(1000);
CommittingOutputStream cos = new CommittingOutputStream();
setupStreamProvider(passed, baos, cos);
checkNotYetCommitted(passed, baos, cos);
cos.write((byte) 1);
checkCommitted(passed, cos);
check(baos, new byte[]{1});
cos.write((byte) 2);
checkCommitted(passed, cos);
cos.write((byte) 3);
cos.write((byte) 4);
check(baos, new byte[]{1, 2, 3, 4});
cos.write((byte) 5);
check(baos, new byte[]{1, 2, 3, 4, 5});
cos.commit();
checkCommitted(passed, cos);
check(baos, new byte[]{1, 2, 3, 4, 5});
cos.close();
}
private void check(ByteArrayOutputStream baos, byte... bytes) {
assertEquals(bytes == null ? 0 : bytes.length, baos.size());
if (bytes != null) {
final byte[] actualBytes = baos.toByteArray();
for (int i = 0; i < bytes.length; i++) {
assertEquals(bytes[i], actualBytes[i]);
}
}
}
@Test
public void testPropertiesWithMessageContext() throws IOException {
final int size = 20;
Map<String, Object> properties = new HashMap<>();
properties.put(CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER, size);
final RuntimeType runtime = RuntimeType.CLIENT;
checkBufferSize(size, properties, runtime);
}
@Test
public void testPropertiesWithMessageContextVeryBigBuffer() throws IOException {
final int size = 200000;
Map<String, Object> properties = new HashMap<>();
properties.put(CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER, size);
final RuntimeType runtime = RuntimeType.CLIENT;
checkBufferSize(size, properties, runtime);
}
@Test
public void testPropertiesWithMessageContextMissingServerSpecific() throws IOException {
final int size = 22;
Map<String, Object> properties = new HashMap<>();
properties.put(CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER, size);
properties.put(CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER + ".client", size * 2);
checkBufferSize(size, properties, RuntimeType.SERVER);
}
@Test
public void testPropertiesWithMessageContextMissingServerAtAll() throws IOException {
final int size = 22;
Map<String, Object> properties = new HashMap<>();
properties.put(PropertiesHelper.getPropertyNameForRuntime(
CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER, RuntimeType.CLIENT), size);
checkBufferSize(CommittingOutputStream.DEFAULT_BUFFER_SIZE, properties, RuntimeType.SERVER);
checkBufferSize(size, properties, RuntimeType.CLIENT);
}
@Test
public void testPropertiesWithMessageContextClientOverrides() throws IOException {
final int size = 22;
Map<String, Object> properties = new HashMap<>();
properties.put(CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER, size);
properties.put(PropertiesHelper.getPropertyNameForRuntime(CommonProperties.OUTBOUND_CONTENT_LENGTH_BUFFER,
RuntimeType.CLIENT), size * 2);
checkBufferSize(size * 2, properties, RuntimeType.CLIENT);
checkBufferSize(size, properties, RuntimeType.SERVER);
}
@Test
public void testPropertiesWithMessageContextDefaultNoProps() throws IOException {
Map<String, Object> properties = new HashMap<>();
final RuntimeType runtime = RuntimeType.CLIENT;
checkBufferSize(CommittingOutputStream.DEFAULT_BUFFER_SIZE, properties, runtime);
}
private void checkBufferSize(int expectedSize, Map<String, Object> properties, RuntimeType runtime) throws IOException {
OutboundMessageContext outboundMessageContext = new OutboundMessageContext((Configuration) null);
final Passed passed = new Passed();
outboundMessageContext.setStreamProvider(new OutboundMessageContext.StreamProvider() {
@Override
public OutputStream getOutputStream(int contentLength) throws IOException {
assertEquals(-1, contentLength);
passed.pass();
return null;
}
});
CommonConfig configuration = new CommonConfig(runtime, ComponentBag.INCLUDE_ALL);
configuration.setProperties(properties);
outboundMessageContext.enableBuffering(configuration);
final OutputStream entityStream = outboundMessageContext.getEntityStream();
for (int i = 1; (i < 1000000) && (!passed.b); i++) {
entityStream.write((byte) 65);
if (i >= expectedSize + 1) {
break;
} else {
assertFalse(passed.b, "committed already with byte #" + i);
}
}
assertTrue(passed.b);
}
@Test
public void testEnableBuffering() {
CommittingOutputStream cos = new CommittingOutputStream();
cos.enableBuffering(500);
}
@Test
public void testEnableBufferingIllegalStateException() throws IOException {
CommittingOutputStream cos = new CommittingOutputStream();
cos.setStreamProvider(new OutboundMessageContext.StreamProvider() {
@Override
public OutputStream getOutputStream(int contentLength) throws IOException {
return null;
}
});
cos.write('a');
try {
cos.enableBuffering(500);
fail("should throw IllegalStateException because of late setup of enableBuffering when bytes are already written.");
} catch (IllegalStateException e) {
System.out.println("this is ok - exception should be thrown: " + e.getMessage());
// ok - should be thrown (bytes are already written).
}
}
@Test
public void testSetStramProviderIllegalStateException1() throws IOException {
CommittingOutputStream cos = new CommittingOutputStream();
cos.enableBuffering(1);
writeAndCheckIllegalState(cos);
}
@Test
public void testSetStramProviderIllegalStateException2() throws IOException {
CommittingOutputStream cos = new CommittingOutputStream();
writeAndCheckIllegalState(cos);
}
private void writeAndCheckIllegalState(CommittingOutputStream cos) throws IOException {
try {
cos.write('a');
cos.write('a');
cos.write('a');
cos.write('a');
cos.write('a');
fail("should throw IllegalStateException because of missing stream provider (bytes are already written).");
} catch (IllegalStateException e) {
System.out.println("this is ok - exception should be thrown: " + e.getMessage());
// ok - should be thrown (bytes are already written).
}
}
}