8294193: Files.createDirectories throws FileAlreadyExistsException for a symbolic link whose target is an existing directory

Reviewed-by: alanb
This commit is contained in:
Jaikiran Pai 2022-09-26 05:16:31 +00:00
parent 3675f4c2af
commit 169a5d48af
3 changed files with 127 additions and 37 deletions

View File

@ -805,7 +805,7 @@ public final class Files {
try {
createDirectory(dir, attrs);
} catch (FileAlreadyExistsException x) {
if (!isDirectory(dir, LinkOption.NOFOLLOW_LINKS))
if (!isDirectory(dir))
throw x;
}
}

View File

@ -0,0 +1,124 @@
/*
* Copyright (c) 2022, 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.
*/
import java.io.IOException;
import java.nio.file.FileAlreadyExistsException;
import java.nio.file.Files;
import java.nio.file.Path;
import static org.testng.Assert.*;
import org.testng.SkipException;
import org.testng.annotations.Test;
/*
* @test
* @bug 8032220 8293792
* @summary Test java.nio.file.Files.createDirectories method
* @library ..
* @run testng CreateDirectories
*/
public class CreateDirectories {
/**
* Test Files.createDirectories symbolic file with an existing directory.
*/
@Test
public void testSymlinkDir() throws Exception {
// create a temp dir as the "root" in which we will run our tests.
final Path top = TestUtil.createTemporaryDirectory();
if (!TestUtil.supportsLinks(top)) {
System.out.println("Skipping tests since symbolic links isn't " +
"supported under directory "+ top);
throw new SkipException("Symbolic links not supported");
}
System.out.println("Running tests under directory " + top.toAbsolutePath());
final Path fooDir = Files.createDirectory(top.resolve("foo"));
assertTrue(Files.isDirectory(fooDir),
fooDir + " was expected to be a directory but wasn't");
// now create a symlink to the "foo" dir
final Path symlink = Files.createSymbolicLink(top.resolve("symlinkToFoo"),
fooDir.toAbsolutePath());
assertTrue(Files.isSymbolicLink(symlink),
symlink + " was expected to be a symlink but wasn't");
assertTrue(Files.isDirectory(symlink),
symlink + " was expected to be a directory but wasn't");
// now create a directory under the symlink (which effectively creates a directory under
// "foo")
final Path barDir = Files.createDirectory(symlink.resolve("bar"));
assertTrue(Files.isDirectory(barDir),
barDir + " was expected to be a directory but wasn't");
// ultimately, we now have this directory structure:
// <root-dir>
// |--- foo
// | |--- bar
// |
// |--- symlinkToFoo -> (links to) <absolute-path-to-root-dir>/foo
// now call Files.createDirectories on each of these existing directory/symlink paths
// and expect each one to succeed
Files.createDirectories(fooDir); // ./<root-dir>/foo
Files.createDirectories(symlink); // ./<root-dir>/symlinkToFoo
Files.createDirectories(barDir); // ./<root-dir>/symlinkToFoo/bar
}
/**
* Tests Files.createDirectories
*/
@Test
public void testCreateDirectories() throws IOException {
final Path tmpdir = TestUtil.createTemporaryDirectory();
// a no-op
Files.createDirectories(tmpdir);
// create one directory
Path subdir = tmpdir.resolve("a");
Files.createDirectories(subdir);
assertTrue(Files.isDirectory(subdir), subdir + " was expected to be a directory," +
" but wasn't");
// create parents
subdir = subdir.resolve("b/c/d");
Files.createDirectories(subdir);
assertTrue(Files.isDirectory(subdir), subdir + " was expected to be a directory," +
" but wasn't");
// existing file is not a directory
Path file = Files.createFile(tmpdir.resolve("x"));
try {
Files.createDirectories(file);
throw new RuntimeException("failure expected");
} catch (FileAlreadyExistsException x) { }
try {
Files.createDirectories(file.resolve("y"));
throw new RuntimeException("failure expected");
} catch (IOException x) { }
// the root directory always exists
Path root = Path.of("/");
Files.createDirectories(root);
Files.createDirectories(root.toAbsolutePath());
}
}

View File

@ -22,8 +22,8 @@
*/
/* @test
* @bug 4313887 6838333 8005566 8032220 8215467 8255576 8286160
* @summary Unit test for miscellenous methods in java.nio.file.Files
* @bug 4313887 6838333 8005566 8215467 8255576 8286160
* @summary Unit test for miscellaneous methods in java.nio.file.Files
* @library .. /test/lib
* @build jdk.test.lib.Platform
* @run main Misc
@ -51,7 +51,6 @@ public class Misc {
public static void main(String[] args) throws IOException {
Path dir = TestUtil.createTemporaryDirectory();
try {
testCreateDirectories(dir);
testIsHidden(dir);
testIsSameFile(dir);
testFileTypeMethods(dir);
@ -61,39 +60,6 @@ public class Misc {
}
}
/**
* Tests createDirectories
*/
static void testCreateDirectories(Path tmpdir) throws IOException {
// a no-op
createDirectories(tmpdir);
// create one directory
Path subdir = tmpdir.resolve("a");
createDirectories(subdir);
assertTrue(exists(subdir));
// create parents
subdir = subdir.resolve("b/c/d");
createDirectories(subdir);
assertTrue(exists(subdir));
// existing file is not a directory
Path file = createFile(tmpdir.resolve("x"));
try {
createDirectories(file);
throw new RuntimeException("failure expected");
} catch (FileAlreadyExistsException x) { }
try {
createDirectories(file.resolve("y"));
throw new RuntimeException("failure expected");
} catch (IOException x) { }
// the root directory always exists
Path root = Path.of("/");
Files.createDirectories(root);
Files.createDirectories(root.toAbsolutePath());
}
/**
* Tests isHidden