6b396da278
Reviewed-by: alanb
438 lines
16 KiB
Java
438 lines
16 KiB
Java
/*
|
|
* Copyright (c) 2008, 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 4313887 6838333 8062795
|
|
* @summary Unit test for java.nio.file.attribute.PosixFileAttributeView
|
|
* @library ../..
|
|
*/
|
|
|
|
import java.nio.file.*;
|
|
import static java.nio.file.LinkOption.*;
|
|
import java.nio.file.attribute.*;
|
|
import java.io.IOException;
|
|
import java.util.*;
|
|
|
|
/**
|
|
* Unit test for PosixFileAttributeView, passing silently if this attribute
|
|
* view is not available.
|
|
*/
|
|
|
|
public class Basic {
|
|
|
|
/**
|
|
* Use view to update permission to the given mode and check that the
|
|
* permissions have been updated.
|
|
*/
|
|
static void testPermissions(Path file, String mode) throws IOException {
|
|
System.out.format("change mode: %s\n", mode);
|
|
Set<PosixFilePermission> perms = PosixFilePermissions.fromString(mode);
|
|
|
|
// change permissions and re-read them.
|
|
Files.setPosixFilePermissions(file, perms);
|
|
Set<PosixFilePermission> current = Files.getPosixFilePermissions(file);
|
|
if (!current.equals(perms)) {
|
|
throw new RuntimeException("Actual permissions: " +
|
|
PosixFilePermissions.toString(current) + ", expected: " +
|
|
PosixFilePermissions.toString(perms));
|
|
}
|
|
|
|
// repeat test using setAttribute/getAttribute
|
|
Files.setAttribute(file, "posix:permissions", perms);
|
|
current = (Set<PosixFilePermission>)Files.getAttribute(file, "posix:permissions");
|
|
if (!current.equals(perms)) {
|
|
throw new RuntimeException("Actual permissions: " +
|
|
PosixFilePermissions.toString(current) + ", expected: " +
|
|
PosixFilePermissions.toString(perms));
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Check that the actual permissions of a file match or make it more
|
|
* secure than requested
|
|
*/
|
|
static void checkSecure(Set<PosixFilePermission> requested,
|
|
Set<PosixFilePermission> actual)
|
|
{
|
|
for (PosixFilePermission perm: actual) {
|
|
if (!requested.contains(perm)) {
|
|
throw new RuntimeException("Actual permissions: " +
|
|
PosixFilePermissions.toString(actual) + ", requested: " +
|
|
PosixFilePermissions.toString(requested) +
|
|
" - file is less secure than requested");
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Create file with given mode and check that the file is created with a
|
|
* mode that is not less secure
|
|
*/
|
|
static void createWithPermissions(Path file,
|
|
String mode)
|
|
throws IOException
|
|
{
|
|
Set<PosixFilePermission> requested = PosixFilePermissions.fromString(mode);
|
|
FileAttribute<Set<PosixFilePermission>> attr =
|
|
PosixFilePermissions.asFileAttribute(requested);
|
|
System.out.format("create file with mode: %s\n", mode);
|
|
Files.createFile(file, attr);
|
|
try {
|
|
checkSecure(requested,
|
|
Files.getFileAttributeView(file, PosixFileAttributeView.class)
|
|
.readAttributes()
|
|
.permissions());
|
|
} finally {
|
|
Files.delete(file);
|
|
}
|
|
|
|
System.out.format("create directory with mode: %s\n", mode);
|
|
Files.createDirectory(file, attr);
|
|
try {
|
|
checkSecure(requested,
|
|
Files.getFileAttributeView(file, PosixFileAttributeView.class)
|
|
.readAttributes()
|
|
.permissions());
|
|
} finally {
|
|
Files.delete(file);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Test the setPermissions/permissions methods.
|
|
*/
|
|
static void permissionTests(Path dir)
|
|
throws IOException
|
|
{
|
|
System.out.println("-- Permission Tests --");
|
|
|
|
// create file and test updating and reading its permissions
|
|
Path file = dir.resolve("foo");
|
|
System.out.format("create %s\n", file);
|
|
Files.createFile(file);
|
|
try {
|
|
// get initial permissions so that we can restore them later
|
|
PosixFileAttributeView view =
|
|
Files.getFileAttributeView(file, PosixFileAttributeView.class);
|
|
Set<PosixFilePermission> save = view.readAttributes()
|
|
.permissions();
|
|
|
|
// test various modes
|
|
try {
|
|
testPermissions(file, "---------");
|
|
testPermissions(file, "r--------");
|
|
testPermissions(file, "-w-------");
|
|
testPermissions(file, "--x------");
|
|
testPermissions(file, "rwx------");
|
|
testPermissions(file, "---r-----");
|
|
testPermissions(file, "----w----");
|
|
testPermissions(file, "-----x---");
|
|
testPermissions(file, "---rwx---");
|
|
testPermissions(file, "------r--");
|
|
testPermissions(file, "-------w-");
|
|
testPermissions(file, "--------x");
|
|
testPermissions(file, "------rwx");
|
|
testPermissions(file, "r--r-----");
|
|
testPermissions(file, "r--r--r--");
|
|
testPermissions(file, "rw-rw----");
|
|
testPermissions(file, "rwxrwx---");
|
|
testPermissions(file, "rw-rw-r--");
|
|
testPermissions(file, "r-xr-x---");
|
|
testPermissions(file, "r-xr-xr-x");
|
|
testPermissions(file, "rwxrwxrwx");
|
|
} finally {
|
|
view.setPermissions(save);
|
|
}
|
|
} finally {
|
|
Files.delete(file);
|
|
}
|
|
|
|
if (TestUtil.supportsLinks(dir)) {
|
|
// create link (to file that doesn't exist) and test reading of
|
|
// permissions
|
|
Path link = dir.resolve("link");
|
|
System.out.format("create link %s\n", link);
|
|
Files.createSymbolicLink(link, file);
|
|
try {
|
|
PosixFileAttributes attrs =
|
|
Files.getFileAttributeView(link,
|
|
PosixFileAttributeView.class,
|
|
NOFOLLOW_LINKS)
|
|
.readAttributes();
|
|
if (!attrs.isSymbolicLink()) {
|
|
throw new RuntimeException("not a link");
|
|
}
|
|
} finally {
|
|
Files.delete(link);
|
|
}
|
|
|
|
// test that setting permissions on paths with and without
|
|
// links succeeds when the NOFOLLOW_LINKS option is set
|
|
|
|
// ensure there are no links in the path to test
|
|
Path realDir = dir.toRealPath();
|
|
|
|
// realDir/a/b/c/d
|
|
Path leaf = realDir.resolve(Path.of("a", "b", "c", "d"));
|
|
Files.createDirectories(leaf);
|
|
|
|
// realDir/a/b/c/d/FUBAR
|
|
Path sansLinks = Files.createTempFile(leaf, "FU", "BAR");
|
|
|
|
PosixFileAttributeView sansView =
|
|
Files.getFileAttributeView(sansLinks,
|
|
PosixFileAttributeView.class,
|
|
LinkOption.NOFOLLOW_LINKS);
|
|
sansView.setPermissions(Set.of(PosixFilePermission.OWNER_WRITE));
|
|
sansView.setPermissions(Set.of(PosixFilePermission.OWNER_WRITE));
|
|
|
|
// reinstate read permission
|
|
sansView.setPermissions(Set.of(PosixFilePermission.OWNER_READ,
|
|
PosixFilePermission.OWNER_WRITE));
|
|
|
|
// lien -> realDir/a/b/c
|
|
Path lien = realDir.resolve(Path.of("a", "lien"));
|
|
Files.createSymbolicLink(lien,
|
|
realDir.resolve(Path.of("a", "b", "c")));
|
|
|
|
// lien/d/FUBAR
|
|
Path withLinks = lien.resolve(Path.of("d"),
|
|
sansLinks.getFileName());
|
|
|
|
PosixFileAttributeView withView =
|
|
Files.getFileAttributeView(withLinks,
|
|
PosixFileAttributeView.class,
|
|
LinkOption.NOFOLLOW_LINKS);
|
|
withView.setPermissions(Set.of(PosixFilePermission.OWNER_WRITE));
|
|
withView.setPermissions(Set.of(PosixFilePermission.OWNER_WRITE));
|
|
}
|
|
|
|
System.out.println("OKAY");
|
|
}
|
|
|
|
/**
|
|
* Test creating a file and directory with initial permissios
|
|
*/
|
|
static void createTests(Path dir)
|
|
throws IOException
|
|
{
|
|
System.out.println("-- Create Tests --");
|
|
|
|
Path file = dir.resolve("foo");
|
|
|
|
createWithPermissions(file, "---------");
|
|
createWithPermissions(file, "r--------");
|
|
createWithPermissions(file, "-w-------");
|
|
createWithPermissions(file, "--x------");
|
|
createWithPermissions(file, "rwx------");
|
|
createWithPermissions(file, "---r-----");
|
|
createWithPermissions(file, "----w----");
|
|
createWithPermissions(file, "-----x---");
|
|
createWithPermissions(file, "---rwx---");
|
|
createWithPermissions(file, "------r--");
|
|
createWithPermissions(file, "-------w-");
|
|
createWithPermissions(file, "--------x");
|
|
createWithPermissions(file, "------rwx");
|
|
createWithPermissions(file, "r--r-----");
|
|
createWithPermissions(file, "r--r--r--");
|
|
createWithPermissions(file, "rw-rw----");
|
|
createWithPermissions(file, "rwxrwx---");
|
|
createWithPermissions(file, "rw-rw-r--");
|
|
createWithPermissions(file, "r-xr-x---");
|
|
createWithPermissions(file, "r-xr-xr-x");
|
|
createWithPermissions(file, "rwxrwxrwx");
|
|
|
|
System.out.println("OKAY");
|
|
}
|
|
|
|
/**
|
|
* Test setOwner/setGroup methods - this test simply exercises the
|
|
* methods to avoid configuration.
|
|
*/
|
|
static void ownerTests(Path dir)
|
|
throws IOException
|
|
{
|
|
System.out.println("-- Owner Tests --");
|
|
|
|
Path file = dir.resolve("gus");
|
|
System.out.format("create %s\n", file);
|
|
|
|
Files.createFile(file);
|
|
try {
|
|
|
|
// read attributes of directory to get owner/group
|
|
PosixFileAttributeView view =
|
|
Files.getFileAttributeView(file, PosixFileAttributeView.class);
|
|
PosixFileAttributes attrs = view.readAttributes();
|
|
|
|
// set to existing owner/group
|
|
view.setOwner(attrs.owner());
|
|
view.setGroup(attrs.group());
|
|
|
|
// repeat test using set/getAttribute
|
|
UserPrincipal owner = (UserPrincipal)Files.getAttribute(file, "posix:owner");
|
|
Files.setAttribute(file, "posix:owner", owner);
|
|
UserPrincipal group = (UserPrincipal)Files.getAttribute(file, "posix:group");
|
|
Files.setAttribute(file, "posix:group", group);
|
|
|
|
} finally {
|
|
Files.delete(file);
|
|
}
|
|
|
|
System.out.println("OKAY");
|
|
}
|
|
|
|
/**
|
|
* Test the lookupPrincipalByName/lookupPrincipalByGroupName methods
|
|
*/
|
|
static void lookupPrincipalTests(Path dir)
|
|
throws IOException
|
|
{
|
|
System.out.println("-- Lookup UserPrincipal Tests --");
|
|
|
|
UserPrincipalLookupService lookupService = dir.getFileSystem()
|
|
.getUserPrincipalLookupService();
|
|
|
|
// read attributes of directory to get owner/group
|
|
PosixFileAttributes attrs = Files.readAttributes(dir, PosixFileAttributes.class);
|
|
|
|
// lookup owner and check it matches file's owner
|
|
System.out.format("lookup: %s\n", attrs.owner().getName());
|
|
try {
|
|
UserPrincipal owner = lookupService.lookupPrincipalByName(attrs.owner().getName());
|
|
if (owner instanceof GroupPrincipal)
|
|
throw new RuntimeException("owner is a group?");
|
|
if (!owner.equals(attrs.owner()))
|
|
throw new RuntimeException("owner different from file owner");
|
|
} catch (UserPrincipalNotFoundException x) {
|
|
System.out.println("user not found - test skipped");
|
|
}
|
|
|
|
// lookup group and check it matches file's group-owner
|
|
System.out.format("lookup group: %s\n", attrs.group().getName());
|
|
try {
|
|
GroupPrincipal group = lookupService.lookupPrincipalByGroupName(attrs.group().getName());
|
|
if (!group.equals(attrs.group()))
|
|
throw new RuntimeException("group different from file group-owner");
|
|
} catch (UserPrincipalNotFoundException x) {
|
|
System.out.println("group not found - test skipped");
|
|
}
|
|
|
|
// test that UserPrincipalNotFoundException is thrown
|
|
String invalidPrincipal = "scumbag99";
|
|
try {
|
|
System.out.format("lookup: %s\n", invalidPrincipal);
|
|
lookupService.lookupPrincipalByName(invalidPrincipal);
|
|
throw new RuntimeException("'" + invalidPrincipal + "' is a valid user?");
|
|
} catch (UserPrincipalNotFoundException x) {
|
|
}
|
|
try {
|
|
System.out.format("lookup group: %s\n", invalidPrincipal);
|
|
lookupService.lookupPrincipalByGroupName("idonotexist");
|
|
throw new RuntimeException("'" + invalidPrincipal + "' is a valid group?");
|
|
} catch (UserPrincipalNotFoundException x) {
|
|
}
|
|
System.out.println("OKAY");
|
|
}
|
|
|
|
/**
|
|
* Test various exceptions are thrown as expected
|
|
*/
|
|
@SuppressWarnings("unchecked")
|
|
static void exceptionsTests(Path dir)
|
|
throws IOException
|
|
{
|
|
System.out.println("-- Exceptions --");
|
|
|
|
PosixFileAttributeView view =
|
|
Files.getFileAttributeView(dir,PosixFileAttributeView.class);
|
|
|
|
// NullPointerException
|
|
try {
|
|
view.setOwner(null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
}
|
|
try {
|
|
view.setGroup(null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
}
|
|
|
|
UserPrincipalLookupService lookupService = dir.getFileSystem()
|
|
.getUserPrincipalLookupService();
|
|
try {
|
|
lookupService.lookupPrincipalByName(null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
}
|
|
try {
|
|
lookupService.lookupPrincipalByGroupName(null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
}
|
|
try {
|
|
view.setPermissions(null);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
}
|
|
try {
|
|
Set<PosixFilePermission> perms = new HashSet<>();
|
|
perms.add(null);
|
|
view.setPermissions(perms);
|
|
throw new RuntimeException("NullPointerException not thrown");
|
|
} catch (NullPointerException x) {
|
|
}
|
|
|
|
// ClassCastException
|
|
try {
|
|
Set perms = new HashSet(); // raw type
|
|
perms.add(new Object());
|
|
view.setPermissions(perms);
|
|
throw new RuntimeException("ClassCastException not thrown");
|
|
} catch (ClassCastException x) {
|
|
}
|
|
|
|
System.out.println("OKAY");
|
|
}
|
|
|
|
public static void main(String[] args) throws IOException {
|
|
Path dir = TestUtil.createTemporaryDirectory();
|
|
try {
|
|
if (!Files.getFileStore(dir).supportsFileAttributeView("posix")) {
|
|
System.out.println("PosixFileAttributeView not supported");
|
|
return;
|
|
}
|
|
|
|
permissionTests(dir);
|
|
createTests(dir);
|
|
ownerTests(dir);
|
|
lookupPrincipalTests(dir);
|
|
exceptionsTests(dir);
|
|
|
|
} finally {
|
|
TestUtil.removeAll(dir);
|
|
}
|
|
}
|
|
}
|