Skip to content

Commit 8be1f9e

Browse files
committed
Create delegate on InflaterInputStream and DeflaterOutputStream to call end on Inflater and Deflater
Signed-off-by: Tanguy Lambert <[email protected]> Add missing import Add missing Copyright comments Signed-off-by: Tanguy LAMBERT <[email protected]>
1 parent ff5f5a8 commit 8be1f9e

File tree

3 files changed

+183
-2
lines changed

3 files changed

+183
-2
lines changed

core-common/src/main/java/org/glassfish/jersey/message/DeflateEncoder.java

+4-2
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,8 @@
3131

3232
import javax.inject.Inject;
3333

34+
import org.glassfish.jersey.message.internal.ClosingDeflaterOutputStream;
35+
import org.glassfish.jersey.message.internal.ClosingInflaterInputStream;
3436
import org.glassfish.jersey.spi.ContentEncoder;
3537

3638
/**
@@ -77,7 +79,7 @@ public InputStream decode(String contentEncoding, InputStream encodedStream)
7779
return new InflaterInputStream(markSupportingStream);
7880
} else {
7981
// no zlib wrapper
80-
return new InflaterInputStream(markSupportingStream, new Inflater(true));
82+
return new ClosingInflaterInputStream(markSupportingStream, true);
8183
}
8284
}
8385

@@ -98,7 +100,7 @@ public OutputStream encode(String contentEncoding, OutputStream entityStream)
98100
}
99101

100102
return deflateWithoutZLib
101-
? new DeflaterOutputStream(entityStream, new Deflater(Deflater.DEFAULT_COMPRESSION, true))
103+
? new ClosingDeflaterOutputStream(entityStream, Deflater.DEFAULT_COMPRESSION, true)
102104
: new DeflaterOutputStream(entityStream);
103105
}
104106
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,83 @@
1+
/*
2+
* Copyright (c) 2024 Oracle and/or its affiliates. All rights reserved.
3+
*
4+
* This program and the accompanying materials are made available under the
5+
* terms of the Eclipse Public License v. 2.0, which is available at
6+
* http://www.eclipse.org/legal/epl-2.0.
7+
*
8+
* This Source Code may also be made available under the following Secondary
9+
* Licenses when the conditions for such availability set forth in the
10+
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
11+
* version 2 with the GNU Classpath Exception, which is available at
12+
* https://www.gnu.org/software/classpath/license.html.
13+
*
14+
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
15+
*/
16+
17+
package org.glassfish.jersey.message.internal;
18+
19+
import java.io.IOException;
20+
import java.io.OutputStream;
21+
import java.util.zip.Deflater;
22+
import java.util.zip.DeflaterOutputStream;
23+
24+
/**
25+
* After Java 9, in the class Deflater the method finalize doesn't call end anymore
26+
* The method end must be called explicitly. But when using the constructor
27+
* DeflaterOutputStream(OutputStream out, Deflater def), the Deflater.end() method will
28+
* not be called when closing the stream.
29+
* This lead to memory leaks in the off-heap.
30+
*
31+
* @see java.util.zip.Deflater
32+
* @see java.util.zip.DeflaterOutputStream
33+
*/
34+
public final class ClosingDeflaterOutputStream extends OutputStream {
35+
36+
private final Deflater deflater;
37+
38+
private final DeflaterOutputStream delegate;
39+
40+
private boolean closed = false;
41+
42+
public ClosingDeflaterOutputStream(OutputStream out, int level, boolean nowrap) {
43+
deflater = new Deflater(level, nowrap);
44+
delegate = new DeflaterOutputStream(out, deflater);
45+
}
46+
47+
@Override
48+
public void write(int b) throws IOException {
49+
delegate.write(b);
50+
}
51+
52+
@Override
53+
public void write(byte[] b, int off, int len) throws IOException {
54+
delegate.write(b, off, len);
55+
}
56+
57+
public void finish() throws IOException {
58+
delegate.finish();
59+
}
60+
61+
@Override
62+
public void close() throws IOException {
63+
if (!closed) {
64+
try {
65+
delegate.close();
66+
} finally {
67+
deflater.end();
68+
}
69+
closed = true;
70+
}
71+
}
72+
73+
@Override
74+
public void flush() throws IOException {
75+
delegate.flush();
76+
}
77+
78+
@Override
79+
public void write(byte[] b) throws IOException {
80+
delegate.write(b);
81+
}
82+
83+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
/*
2+
* Copyright (c) 2024 Oracle and/or its affiliates. All rights reserved.
3+
*
4+
* This program and the accompanying materials are made available under the
5+
* terms of the Eclipse Public License v. 2.0, which is available at
6+
* http://www.eclipse.org/legal/epl-2.0.
7+
*
8+
* This Source Code may also be made available under the following Secondary
9+
* Licenses when the conditions for such availability set forth in the
10+
* Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
11+
* version 2 with the GNU Classpath Exception, which is available at
12+
* https://www.gnu.org/software/classpath/license.html.
13+
*
14+
* SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
15+
*/
16+
17+
package org.glassfish.jersey.message.internal;
18+
19+
import java.io.IOException;
20+
import java.io.InputStream;
21+
import java.util.zip.Inflater;
22+
import java.util.zip.InflaterInputStream;
23+
24+
/**
25+
* After Java 9, in the class Inflater the method finalize doesn't call end anymore
26+
* The method end must be called explicitly. But when using the constructor
27+
* InflaterInputStream(InputStream out, Inflater def), the Inflater.end() method will
28+
* not be called when closing the stream.
29+
* This lead to memory leaks in the off-heap.
30+
*
31+
* @see java.util.zip.Inflater
32+
* @see java.util.zip.InflaterInputStream
33+
*/
34+
public class ClosingInflaterInputStream extends InputStream {
35+
36+
private final Inflater inflater;
37+
38+
private final InflaterInputStream delegate;
39+
40+
private boolean closed = false;
41+
42+
public ClosingInflaterInputStream(InputStream inputStream, boolean nowrap) {
43+
inflater = new Inflater(nowrap);
44+
delegate = new InflaterInputStream(inputStream, inflater);
45+
}
46+
47+
@Override
48+
public int read() throws IOException {
49+
return delegate.read();
50+
}
51+
52+
@Override
53+
public int read(byte[] b, int off, int len) throws IOException {
54+
return delegate.read(b, off, len);
55+
}
56+
57+
@Override
58+
public int available() throws IOException {
59+
return delegate.available();
60+
}
61+
62+
@Override
63+
public long skip(long n) throws IOException {
64+
return delegate.skip(n);
65+
}
66+
67+
@Override
68+
public void close() throws IOException {
69+
if (!closed) {
70+
inflater.end();
71+
delegate.close();
72+
closed = true;
73+
}
74+
}
75+
76+
@Override
77+
public boolean markSupported() {
78+
return delegate.markSupported();
79+
}
80+
81+
@Override
82+
public void mark(int readlimit) {
83+
delegate.mark(readlimit);
84+
}
85+
86+
@Override
87+
public void reset() throws IOException {
88+
delegate.reset();
89+
}
90+
91+
@Override
92+
public int read(byte[] b) throws IOException {
93+
return delegate.read(b);
94+
}
95+
96+
}

0 commit comments

Comments
 (0)