jdk-24/test/jdk/sun/net/www/protocol/jar/FileURLConnectionLeak.java
Jesse Glick 9f98136c3a 6956385: URLConnection.getLastModified() leaks file handles for jar:file and file: URLs
Define FileURLConnection.closeInputStream for use by JarURLInputStream.close.
JarURLConnection properly tracks any InputStream it itself opened,
and correspondingly closes the JarFile if necessary (when caches are disabled).
But if its underlying FileURLConnection was used to retrieve a header field,
that would have caused a FileInputStream to be opened
which never gets closed until it is garbage collected.
This means that an application which calls certain methods
on jar:file:/…something.jar!/… URLs will leak file handles,
even if URLConnection caches are supposed to be turned off.
This can delay release of system resources,
and on Windows can prevent the JAR file from being deleted
even after it is no longer in use (for example after URLClassLoader.close).

Reviewed-by: dfuchs, michaelm
2023-06-28 15:33:37 +00:00

72 lines
3.0 KiB
Java

/*
* 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 6956385
* @summary JarURLConnection may fail to close its underlying FileURLConnection
* @run main/othervm FileURLConnectionLeak
*/
import java.net.URI;
import java.net.URLConnection;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.jar.Attributes;
import java.util.jar.JarOutputStream;
import java.util.jar.Manifest;
public class FileURLConnectionLeak {
public static void main(String[] args) throws Exception {
URLConnection.setDefaultUseCaches("file", false);
URLConnection.setDefaultUseCaches("jar", false);
var jar = Path.of("x.jar").toAbsolutePath();
var mani = new Manifest();
mani.getMainAttributes().put(Attributes.Name.MANIFEST_VERSION, "1.0");
try (var os = Files.newOutputStream(jar); var jos = new JarOutputStream(os, mani)) {}
var u = URI.create("jar:" + jar.toUri() + "!/META-INF/MANIFEST.MF").toURL();
// FileURLConnection.is not used, so was always fine:
try (var is = u.openStream()) {
is.transferTo(System.out);
}
// FileURLConnection.is opened implicitly:
var conn = u.openConnection();
conn.getLastModified();
// Idiom to close URLConnection (cf. JDK-8224095), which must also close the other stream:
conn.getInputStream().close();
var fds = Path.of("/proc/" + ProcessHandle.current().pid() + "/fd");
if (Files.isDirectory(fds)) {
// Linux: verify that x.jar is not open
for (var fd : (Iterable<Path>) Files.list(fds)::iterator) {
if (Files.isSymbolicLink(fd)) {
var file = Files.readSymbolicLink(fd);
if (file.equals(jar)) {
throw new IllegalStateException("Still held open " + jar + " from " + fd);
}
}
}
}
// Windows: verify that mandatory file locks do not prevent deletion
Files.delete(jar);
}
}