diff --git a/src/main/java/org/eclipse/yasson/internal/JsonBinding.java b/src/main/java/org/eclipse/yasson/internal/JsonBinding.java index fb185deb..5451e7ad 100644 --- a/src/main/java/org/eclipse/yasson/internal/JsonBinding.java +++ b/src/main/java/org/eclipse/yasson/internal/JsonBinding.java @@ -12,6 +12,7 @@ package org.eclipse.yasson.internal; +import java.io.IOException; import java.io.InputStream; import java.io.OutputStream; import java.io.Reader; @@ -19,6 +20,7 @@ import java.io.StringWriter; import java.io.Writer; import java.lang.reflect.Type; +import java.nio.CharBuffer; import java.nio.charset.Charset; import java.util.Map; import java.util.Set; @@ -73,7 +75,7 @@ public T fromJson(String str, Type type) throws JsonbException { @Override public T fromJson(Reader reader, Class type) throws JsonbException { - try (JsonParser parser = jsonbContext.getJsonProvider().createParser(reader)) { + try (JsonParser parser = jsonbContext.getJsonProvider().createParser(new CloseSuppressingReader(reader))) { DeserializationContextImpl unmarshaller = new DeserializationContextImpl(jsonbContext); return deserialize(type, parser, unmarshaller); } @@ -81,7 +83,7 @@ public T fromJson(Reader reader, Class type) throws JsonbException { @Override public T fromJson(Reader reader, Type type) throws JsonbException { - try (JsonParser parser = jsonbContext.getJsonProvider().createParser(reader)) { + try (JsonParser parser = jsonbContext.getJsonProvider().createParser(new CloseSuppressingReader(reader))) { DeserializationContextImpl unmarshaller = new DeserializationContextImpl(jsonbContext); return deserialize(type, parser, unmarshaller); } @@ -119,7 +121,7 @@ public T fromJsonStructure(JsonStructure jsonStructure, Type runtimeType) th private JsonParser inputStreamParser(InputStream stream) { return jsonbContext.getJsonParserFactory() - .createParser(stream, + .createParser(new CloseSuppressingInputStream(stream), Charset.forName((String) jsonbContext.getConfig() .getProperty(JsonbConfig.ENCODING).orElse("UTF-8"))); } @@ -145,7 +147,7 @@ public String toJson(Object object, Type type) throws JsonbException { @Override public void toJson(Object object, Writer writer) throws JsonbException { final SerializationContextImpl marshaller = new SerializationContextImpl(jsonbContext); - try (JsonGenerator generator = writerGenerator(writer)) { + try (JsonGenerator generator = writerGenerator(new CloseSuppressingWriter(writer))) { marshaller.marshallWithoutClose(object, generator); } } @@ -153,7 +155,7 @@ public void toJson(Object object, Writer writer) throws JsonbException { @Override public void toJson(Object object, Type type, Writer writer) throws JsonbException { final SerializationContextImpl marshaller = new SerializationContextImpl(jsonbContext, type); - try (JsonGenerator generator = writerGenerator(writer)) { + try (JsonGenerator generator = writerGenerator(new CloseSuppressingWriter(writer))) { marshaller.marshallWithoutClose(object, generator); } } @@ -226,7 +228,7 @@ private JsonGenerator streamGenerator(OutputStream stream) { Map factoryProperties = jsonbContext.createJsonpProperties(jsonbContext.getConfig()); final String encoding = (String) jsonbContext.getConfig().getProperty(JsonbConfig.ENCODING).orElse("UTF-8"); return jsonbContext.getJsonProvider().createGeneratorFactory(factoryProperties) - .createGenerator(stream, Charset.forName(encoding)); + .createGenerator(new CloseSuppressingOutputStream(stream), Charset.forName(encoding)); } @Override @@ -234,4 +236,246 @@ public void close() throws Exception { jsonbContext.getComponentInstanceCreator().close(); } + /** + * {@link OutputStream} that suppresses {@link OutputStream#close()}. + */ + static final class CloseSuppressingOutputStream extends OutputStream { + + private final OutputStream delegate; + + CloseSuppressingOutputStream(OutputStream delegate) { + this.delegate = delegate; + } + + @Override + public void close() { + // suppress + } + + @Override + public void write(int b) throws IOException { + delegate.write(b); + } + + @Override + public void write(byte[] b) throws IOException { + delegate.write(b); + } + + @Override + public void write(byte[] b, int off, int len) throws IOException { + delegate.write(b, off, len); + } + + } + + /** + * {@link InputStream} that suppresses {@link InputStream#close()}. + */ + static final class CloseSuppressingInputStream extends InputStream { + + private final InputStream delegate; + + CloseSuppressingInputStream(InputStream delegate) { + this.delegate = delegate; + } + + @Override + public void close() { + // suppress + } + + @Override + public int read() throws IOException { + return delegate.read(); + } + + @Override + public int read(byte[] b) throws IOException { + return delegate.read(b); + } + + @Override + public int read(byte[] b, int off, int len) throws IOException { + return delegate.read(b, off, len); + } + + @Override + public byte[] readAllBytes() throws IOException { + return delegate.readAllBytes(); + } + + @Override + public byte[] readNBytes(int len) throws IOException { + return delegate.readNBytes(len); + } + + @Override + public int readNBytes(byte[] b, int off, int len) throws IOException { + return delegate.readNBytes(b, off, len); + } + + @Override + public long skip(long n) throws IOException { + return delegate.skip(n); + } + + @Override + public int available() throws IOException { + return delegate.available(); + } + + @Override + public void mark(int readlimit) { + delegate.mark(readlimit); + } + + @Override + public void reset() throws IOException { + delegate.reset(); + } + + @Override + public boolean markSupported() { + return delegate.markSupported(); + } + + @Override + public long transferTo(OutputStream out) throws IOException { + return delegate.transferTo(out); + } + + } + + /** + * {@link Reader} that suppresses {@link Reader#close()}. + */ + static final class CloseSuppressingReader extends Reader { + + private final Reader delegate; + + CloseSuppressingReader(Reader delegate) { + this.delegate = delegate; + } + + @Override + public int read(CharBuffer target) throws IOException { + return delegate.read(target); + } + + @Override + public int read() throws IOException { + return delegate.read(); + } + + @Override + public int read(char[] cbuf) throws IOException { + return delegate.read(cbuf); + } + + @Override + public int read(char[] cbuf, int off, int len) throws IOException { + return delegate.read(cbuf, off, len); + } + + @Override + public long skip(long n) throws IOException { + return delegate.skip(n); + } + + @Override + public boolean ready() throws IOException { + return delegate.ready(); + } + + @Override + public boolean markSupported() { + return delegate.markSupported(); + } + + @Override + public void mark(int readAheadLimit) throws IOException { + delegate.mark(readAheadLimit); + } + + @Override + public void reset() throws IOException { + delegate.reset(); + } + + @Override + public void close() { + // suppress + } + + @Override + public long transferTo(Writer out) throws IOException { + return delegate.transferTo(out); + } + + } + + /** + * {@link Writer} that suppresses {@link Writer#close()}. + */ + static final class CloseSuppressingWriter extends Writer { + + private final Writer delegate; + + CloseSuppressingWriter(Writer delegate) { + this.delegate = delegate; + } + + @Override + public void write(int c) throws IOException { + delegate.write(c); + } + + @Override + public void write(char[] cbuf) throws IOException { + delegate.write(cbuf); + } + + @Override + public void write(char[] cbuf, int off, int len) throws IOException { + delegate.write(cbuf, off, len); + } + + @Override + public void write(String str) throws IOException { + delegate.write(str); + } + + @Override + public void write(String str, int off, int len) throws IOException { + delegate.write(str, off, len); + } + + @Override + public Writer append(CharSequence csq) throws IOException { + return delegate.append(csq); + } + + @Override + public Writer append(CharSequence csq, int start, int end) throws IOException { + return delegate.append(csq, start, end); + } + + @Override + public Writer append(char c) throws IOException { + return delegate.append(c); + } + + @Override + public void flush() throws IOException { + delegate.flush(); + } + + @Override + public void close() { + // suppress + } + + } + } diff --git a/src/test/java/org/eclipse/yasson/Issue586Test.java b/src/test/java/org/eclipse/yasson/Issue586Test.java new file mode 100644 index 00000000..685aa0f8 --- /dev/null +++ b/src/test/java/org/eclipse/yasson/Issue586Test.java @@ -0,0 +1,372 @@ +/* + * Copyright (c) 2024 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, + * or the Eclipse Distribution License v. 1.0 which is available at + * http://www.eclipse.org/org/documents/edl-v10.php. + * + * SPDX-License-Identifier: EPL-2.0 OR BSD-3-Clause + */ + +package org.eclipse.yasson; + +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertNotNull; + +import java.io.ByteArrayInputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.io.Reader; +import java.io.StringReader; +import java.io.Writer; +import java.lang.reflect.Type; +import java.nio.CharBuffer; +import java.nio.charset.StandardCharsets; + +import org.junit.jupiter.api.Test; + +/** + * Verify Jsonb leaves the streams open. + * + * @author Philippe Marschall + */ +public class Issue586Test { + + @Test + public void fromJsonInputStreamClass() { + CloseRememberingInputStream stream = new CloseRememberingInputStream(asInputStream("{\"field\":\"first\"}")); + assertNotNull(Jsonbs.defaultJsonb.fromJson(stream, KeyValue.class)); + assertFalse(stream.isCloseCalled(), "close() should not be called by "); + } + + @Test + public void fromJsonInputStreamType() { + CloseRememberingInputStream stream = new CloseRememberingInputStream(asInputStream("{\"field\":\"first\"}")); + assertNotNull(Jsonbs.defaultJsonb.fromJson(stream, (Type) KeyValue.class)); + assertFalse(stream.isCloseCalled(), "close() should not be called by "); + } + + @Test + public void fromJsonReaderClass() { + CloseRememberingReader stream = new CloseRememberingReader(asReader("{\"field\":\"first\"}")); + assertNotNull(Jsonbs.defaultJsonb.fromJson(stream, KeyValue.class)); + assertFalse(stream.isCloseCalled(), "close() should not be called by "); + } + + @Test + public void fromJsonReaderType() { + CloseRememberingReader reader = new CloseRememberingReader(asReader("{\"field\":\"first\"}")); + assertNotNull(Jsonbs.defaultJsonb.fromJson(reader, (Type) KeyValue.class)); + assertFalse(reader.isCloseCalled(), "close() should not be called by "); + } + @Test + public void toJsonOutputStreamObject() { + CloseRememberingOutputStream stream = new CloseRememberingOutputStream(OutputStream.nullOutputStream()); + Jsonbs.defaultJsonb.toJson(new KeyValue("first"), stream); + assertFalse(stream.isCloseCalled(), "close() should not be called by "); + } + + @Test + public void toJsonOutputStreamObjectAndType() { + CloseRememberingOutputStream stream = new CloseRememberingOutputStream(OutputStream.nullOutputStream()); + Jsonbs.defaultJsonb.toJson(new KeyValue("first"), KeyValue.class, stream); + assertFalse(stream.isCloseCalled(), "close() should not be called by "); + } + + @Test + public void toJsonWriterObject() { + CloseRememberingWriter writer = new CloseRememberingWriter(Writer.nullWriter()); + Jsonbs.defaultJsonb.toJson(new KeyValue("first"), writer); + assertFalse(writer.isCloseCalled(), "close() should not be called by "); + } + + @Test + public void toJsonWriterObjectAndType() { + CloseRememberingWriter writer= new CloseRememberingWriter(Writer.nullWriter()); + Jsonbs.defaultJsonb.toJson(new KeyValue("first"), KeyValue.class, writer); + assertFalse(writer.isCloseCalled(), "close() should not be called by "); + } + + private static InputStream asInputStream(String s) { + return new ByteArrayInputStream(s.getBytes(StandardCharsets.US_ASCII)); + } + + private static Reader asReader(String s) { + return new StringReader(s); + } + + public static class KeyValue { + + public String field; + + public KeyValue() { + + } + + public KeyValue(String field) { + this.field = field; + } + } + + static final class CloseRememberingOutputStream extends OutputStream { + + private final OutputStream delegate; + private boolean closeCalled; + + CloseRememberingOutputStream(OutputStream delegate) { + this.delegate = delegate; + this.closeCalled = false; + } + + @Override + public void close() throws IOException { + closeCalled = true; + delegate.close(); + } + + boolean isCloseCalled() { + return closeCalled; + } + + @Override + public void write(int b) throws IOException { + delegate.write(b); + } + + @Override + public void write(byte[] b) throws IOException { + delegate.write(b); + } + + @Override + public void write(byte[] b, int off, int len) throws IOException { + delegate.write(b, off, len); + } + + } + + static final class CloseRememberingInputStream extends InputStream { + + private final InputStream delegate; + private boolean closeCalled; + + CloseRememberingInputStream(InputStream delegate) { + this.delegate = delegate; + this.closeCalled = false; + } + + @Override + public void close() throws IOException { + closeCalled = true; + delegate.close(); + } + + boolean isCloseCalled() { + return closeCalled; + } + + @Override + public int read() throws IOException { + return delegate.read(); + } + + @Override + public int read(byte[] b) throws IOException { + return delegate.read(b); + } + + @Override + public int read(byte[] b, int off, int len) throws IOException { + return delegate.read(b, off, len); + } + + @Override + public byte[] readAllBytes() throws IOException { + return delegate.readAllBytes(); + } + + @Override + public byte[] readNBytes(int len) throws IOException { + return delegate.readNBytes(len); + } + + @Override + public int readNBytes(byte[] b, int off, int len) throws IOException { + return delegate.readNBytes(b, off, len); + } + + @Override + public long skip(long n) throws IOException { + return delegate.skip(n); + } + + @Override + public int available() throws IOException { + return delegate.available(); + } + + @Override + public void mark(int readlimit) { + delegate.mark(readlimit); + } + + @Override + public void reset() throws IOException { + delegate.reset(); + } + + @Override + public boolean markSupported() { + return delegate.markSupported(); + } + + @Override + public long transferTo(OutputStream out) throws IOException { + return delegate.transferTo(out); + } + + } + + static final class CloseRememberingReader extends Reader { + + private final Reader delegate; + private boolean closeCalled; + + CloseRememberingReader(Reader delegate) { + this.delegate = delegate; + this.closeCalled = false; + } + + @Override + public int read(CharBuffer target) throws IOException { + return delegate.read(target); + } + + @Override + public int read() throws IOException { + return delegate.read(); + } + + @Override + public int read(char[] cbuf) throws IOException { + return delegate.read(cbuf); + } + + @Override + public int read(char[] cbuf, int off, int len) throws IOException { + return delegate.read(cbuf, off, len); + } + + @Override + public long skip(long n) throws IOException { + return delegate.skip(n); + } + + @Override + public boolean ready() throws IOException { + return delegate.ready(); + } + + @Override + public boolean markSupported() { + return delegate.markSupported(); + } + + @Override + public void mark(int readAheadLimit) throws IOException { + delegate.mark(readAheadLimit); + } + + @Override + public void reset() throws IOException { + delegate.reset(); + } + + @Override + public void close() throws IOException { + closeCalled = true; + delegate.close(); + } + + boolean isCloseCalled() { + return closeCalled; + } + + @Override + public long transferTo(Writer out) throws IOException { + return delegate.transferTo(out); + } + + } + + static final class CloseRememberingWriter extends Writer { + + private final Writer delegate; + private boolean closeCalled; + + CloseRememberingWriter(Writer delegate) { + this.delegate = delegate; + this.closeCalled = false; + } + + @Override + public void write(int c) throws IOException { + delegate.write(c); + } + + @Override + public void write(char[] cbuf) throws IOException { + delegate.write(cbuf); + } + + @Override + public void write(char[] cbuf, int off, int len) throws IOException { + delegate.write(cbuf, off, len); + } + + @Override + public void write(String str) throws IOException { + delegate.write(str); + } + + @Override + public void write(String str, int off, int len) throws IOException { + delegate.write(str, off, len); + } + + @Override + public Writer append(CharSequence csq) throws IOException { + return delegate.append(csq); + } + + @Override + public Writer append(CharSequence csq, int start, int end) throws IOException { + return delegate.append(csq, start, end); + } + + @Override + public Writer append(char c) throws IOException { + return delegate.append(c); + } + + @Override + public void flush() throws IOException { + delegate.flush(); + } + + @Override + public void close() throws IOException { + closeCalled = true; + delegate.close(); + } + + boolean isCloseCalled() { + return closeCalled; + } + + } + +}