8136895: Writer not closed with disk full error, file resource leaked
Reviewed-by: djelinski, vtewari
This commit is contained in:
parent
8042a50b99
commit
d55d7e8d87
@ -414,8 +414,8 @@ public final class StreamEncoder extends Writer {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void implClose() throws IOException {
|
void implClose() throws IOException {
|
||||||
flushLeftoverChar(null, true);
|
try (ch; out) {
|
||||||
try {
|
flushLeftoverChar(null, true);
|
||||||
for (;;) {
|
for (;;) {
|
||||||
CoderResult cr = encoder.flush(bb);
|
CoderResult cr = encoder.flush(bb);
|
||||||
if (cr.isUnderflow())
|
if (cr.isUnderflow())
|
||||||
@ -430,15 +430,8 @@ public final class StreamEncoder extends Writer {
|
|||||||
|
|
||||||
if (bb.position() > 0)
|
if (bb.position() > 0)
|
||||||
writeBytes();
|
writeBytes();
|
||||||
if (ch != null)
|
if (out != null)
|
||||||
ch.close();
|
out.flush();
|
||||||
else {
|
|
||||||
try {
|
|
||||||
out.flush();
|
|
||||||
} finally {
|
|
||||||
out.close();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (IOException x) {
|
} catch (IOException x) {
|
||||||
encoder.reset();
|
encoder.reset();
|
||||||
throw x;
|
throw x;
|
||||||
|
@ -0,0 +1,86 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* @test
|
||||||
|
* @bug 8136895
|
||||||
|
* @summary Verify stream closed after write error in StreamEncoder::implClose
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.OutputStream;
|
||||||
|
import java.io.OutputStreamWriter;
|
||||||
|
import java.io.Writer;
|
||||||
|
import java.nio.charset.MalformedInputException;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
|
public class CloseWriterOnFailedFlush {
|
||||||
|
private static final String STR_IOE = "Test"; // IOException
|
||||||
|
private static final String STR_MIE = "\ud83c"; // MalformedInputException
|
||||||
|
|
||||||
|
public static void main(String[] args) throws IOException {
|
||||||
|
boolean failed = false;
|
||||||
|
|
||||||
|
for (String s : new String[] {STR_IOE, STR_MIE}) {
|
||||||
|
System.out.println("string: " + s);
|
||||||
|
ErroringOutputStream stream = new ErroringOutputStream();
|
||||||
|
try (Writer writer = new OutputStreamWriter(stream,
|
||||||
|
StandardCharsets.UTF_8.newEncoder())) {
|
||||||
|
writer.write(s);
|
||||||
|
} catch (IOException ex) {
|
||||||
|
Class exClass = ex.getClass();
|
||||||
|
if (s.equals(STR_IOE) && exClass != IOException.class ||
|
||||||
|
s.equals(STR_MIE) && exClass != MalformedInputException.class)
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stream.isOpen()) {
|
||||||
|
System.err.println("Stream is STILL open");
|
||||||
|
failed = true;
|
||||||
|
} else {
|
||||||
|
System.out.println("Stream is closed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (failed)
|
||||||
|
throw new RuntimeException("Test failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class ErroringOutputStream extends OutputStream {
|
||||||
|
private boolean open = true;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void write(int b) throws IOException {
|
||||||
|
throw new IOException();
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean isOpen() {
|
||||||
|
return open;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close() throws IOException {
|
||||||
|
open = false;
|
||||||
|
System.out.println("Closing");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,88 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* @test
|
||||||
|
* @bug 8136895
|
||||||
|
* @summary Verify channel closed after write error in StreamEncoder::implClose
|
||||||
|
*/
|
||||||
|
|
||||||
|
import java.io.IOException;
|
||||||
|
import java.io.Writer;
|
||||||
|
import java.nio.ByteBuffer;
|
||||||
|
import java.nio.channels.Channels;
|
||||||
|
import java.nio.channels.WritableByteChannel;
|
||||||
|
import java.nio.charset.MalformedInputException;
|
||||||
|
import java.nio.charset.StandardCharsets;
|
||||||
|
|
||||||
|
public class CloseWriterOnFailedFlush {
|
||||||
|
private static final String STR_IOE = "Test"; // IOException
|
||||||
|
private static final String STR_MIE = "\ud83c"; // MalformedInputException
|
||||||
|
|
||||||
|
public static void main(String[] args) throws IOException {
|
||||||
|
boolean failed = false;
|
||||||
|
|
||||||
|
for (String s : new String[] {STR_IOE, STR_MIE}) {
|
||||||
|
System.out.println("string: " + s);
|
||||||
|
ErroringByteChannel channel = new ErroringByteChannel();
|
||||||
|
try (Writer writer = Channels.newWriter
|
||||||
|
(channel, StandardCharsets.UTF_8.newEncoder(), -1 )) {
|
||||||
|
writer.write(s);
|
||||||
|
} catch (IOException ex) {
|
||||||
|
Class exClass = ex.getClass();
|
||||||
|
if (s.equals(STR_IOE) && exClass != IOException.class ||
|
||||||
|
s.equals(STR_MIE) && exClass != MalformedInputException.class)
|
||||||
|
throw ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (channel.isOpen()) {
|
||||||
|
System.err.println("Channel is STILL open");
|
||||||
|
failed = true;
|
||||||
|
} else {
|
||||||
|
System.out.println("Channel is closed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (failed)
|
||||||
|
throw new RuntimeException("Test failed");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static class ErroringByteChannel implements WritableByteChannel {
|
||||||
|
private boolean open = true;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int write(ByteBuffer src) throws IOException {
|
||||||
|
throw new IOException();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isOpen() {
|
||||||
|
return open;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void close() {
|
||||||
|
open = false;
|
||||||
|
System.out.println("Closing");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2008, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2008, 2023, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -31,6 +31,15 @@
|
|||||||
|
|
||||||
import java.io.*;
|
import java.io.*;
|
||||||
public class StreamEncoderClose {
|
public class StreamEncoderClose {
|
||||||
|
private static void ck(String s, int actual, int expected)
|
||||||
|
throws IOException {
|
||||||
|
if (actual != expected) {
|
||||||
|
String msg = String.format("%s: actual (%d) != expected (%d)%n",
|
||||||
|
s, actual, expected);
|
||||||
|
throw new IOException(msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public static void main( String arg[] ) throws Exception {
|
public static void main( String arg[] ) throws Exception {
|
||||||
byte[] expected = {(byte)0x1b,(byte)0x24,(byte)0x42,
|
byte[] expected = {(byte)0x1b,(byte)0x24,(byte)0x42,
|
||||||
(byte)0x30,(byte)0x6c,
|
(byte)0x30,(byte)0x6c,
|
||||||
@ -46,13 +55,10 @@ public class StreamEncoderClose {
|
|||||||
|
|
||||||
//double check, probably not necessary
|
//double check, probably not necessary
|
||||||
byte[] out = baos.toByteArray();
|
byte[] out = baos.toByteArray();
|
||||||
if (out.length != expected.length) {
|
ck("Lengths are unequal", out.length, expected.length);
|
||||||
throw new IOException("Failed");
|
|
||||||
}
|
|
||||||
for (int i = 0; i < out.length; i++) {
|
for (int i = 0; i < out.length; i++) {
|
||||||
//System.out.printf("(byte)0x%x,", out[i] & 0xff);
|
//System.out.printf("(byte)0x%x,", out[i] & 0xff);
|
||||||
if (out[i] != expected[i])
|
ck("Values are unequal", out[i], expected[i]);
|
||||||
throw new IOException("Failed");
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user