alluxio.util.io.FileUtils Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of alluxio-core-common Show documentation
Show all versions of alluxio-core-common Show documentation
Common utilities shared in Alluxio core modules
/*
* The Alluxio Open Foundation licenses this work under the Apache License, version 2.0
* (the "License"). You may not use this work except in compliance with the License, which is
* available at www.apache.org/licenses/LICENSE-2.0
*
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
* either express or implied, as more fully set forth in the License.
*
* See the NOTICE file distributed with this work for information regarding copyright ownership.
*/
package alluxio.util.io;
import alluxio.AlluxioURI;
import alluxio.exception.InvalidPathException;
import alluxio.exception.runtime.AlreadyExistsRuntimeException;
import alluxio.exception.runtime.FailedPreconditionRuntimeException;
import alluxio.exception.runtime.InternalRuntimeException;
import alluxio.exception.runtime.InvalidArgumentRuntimeException;
import alluxio.exception.runtime.NotFoundRuntimeException;
import alluxio.exception.runtime.PermissionDeniedRuntimeException;
import alluxio.exception.runtime.UnimplementedRuntimeException;
import alluxio.exception.runtime.UnknownRuntimeException;
import alluxio.grpc.ErrorType;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Preconditions;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.File;
import java.io.IOException;
import java.nio.ByteBuffer;
import java.nio.file.DirectoryNotEmptyException;
import java.nio.file.FileAlreadyExistsException;
import java.nio.file.FileSystems;
import java.nio.file.FileVisitResult;
import java.nio.file.Files;
import java.nio.file.LinkOption;
import java.nio.file.NoSuchFileException;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.SimpleFileVisitor;
import java.nio.file.StandardCopyOption;
import java.nio.file.attribute.BasicFileAttributes;
import java.nio.file.attribute.GroupPrincipal;
import java.nio.file.attribute.PosixFileAttributeView;
import java.nio.file.attribute.PosixFileAttributes;
import java.nio.file.attribute.PosixFilePermission;
import java.nio.file.attribute.PosixFilePermissions;
import java.nio.file.attribute.UserDefinedFileAttributeView;
import java.nio.file.attribute.UserPrincipal;
import java.nio.file.attribute.UserPrincipalLookupService;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
import javax.annotation.concurrent.ThreadSafe;
/**
* Provides utility methods for working with files and directories.
*
* By convention, methods take file path strings as parameters.
*/
@ThreadSafe
public final class FileUtils {
private static final Logger LOG = LoggerFactory.getLogger(FileUtils.class);
/**
* Changes the local file's group.
*
* @param path that will change owner
* @param group the new group
*/
public static void changeLocalFileGroup(String path, String group) throws IOException {
UserPrincipalLookupService lookupService =
FileSystems.getDefault().getUserPrincipalLookupService();
PosixFileAttributeView view =
Files.getFileAttributeView(Paths.get(path), PosixFileAttributeView.class,
LinkOption.NOFOLLOW_LINKS);
GroupPrincipal groupPrincipal = lookupService.lookupPrincipalByGroupName(group);
view.setGroup(groupPrincipal);
}
/**
* Changes local file's permission.
*
* @param filePath that will change permission
* @param perms the permission, e.g. "rwxr--r--"
*/
public static void changeLocalFilePermission(String filePath, String perms) {
try {
Files.setPosixFilePermissions(Paths.get(filePath), PosixFilePermissions.fromString(perms));
} catch (UnsupportedOperationException e) {
throw new UnimplementedRuntimeException(e, ErrorType.External);
} catch (ClassCastException e) {
throw new InvalidArgumentRuntimeException(e);
} catch (SecurityException e) {
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
throw new UnknownRuntimeException(e);
}
}
/**
* Changes local file's user defined file attribute.
* @param filePath the file to change attribute
* @param name the name of attribute
* @param value the value of attribute
*/
public static void changeLocalFileUserDefinedAttribute(String filePath, String name,
byte[] value) {
try {
UserDefinedFileAttributeView attributeView =
Files.getFileAttributeView(Paths.get(filePath), UserDefinedFileAttributeView.class);
if (attributeView == null) {
throw new UnimplementedRuntimeException(
"UFS does not support getting FileAttributeView with Java.");
}
attributeView.write(name, ByteBuffer.wrap(value));
} catch (UnsupportedOperationException e) {
LOG.warn("UFS does not support getting FileAttributeView with Java.");
throw new UnimplementedRuntimeException(e, ErrorType.External);
} catch (ClassCastException | IllegalArgumentException e) {
String msg = "UFS does not support getting FileAttributeView with Java.";
LOG.warn(msg);
throw new InvalidArgumentRuntimeException(msg, e);
} catch (SecurityException e) {
LOG.warn("UFS does not support getting FileAttributeView with Java.");
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
LOG.warn("UFS does not support getting FileAttributeView with Java.");
throw new UnknownRuntimeException(e);
}
}
/**
* Gets local file's user defined file attribute.
*
* @param filePath
* @return the map of attribute view
*/
public static Map getLocalFileUserDefinedAttribute(String filePath) {
try {
UserDefinedFileAttributeView attributeView =
Files.getFileAttributeView(Paths.get(filePath), UserDefinedFileAttributeView.class);
if (attributeView == null) {
throw new UnimplementedRuntimeException(
"UFS does not support getting FileAttributeView with Java.");
}
Map attrMap = new HashMap<>(attributeView.list().size());
for (String attributeName : attributeView.list()) {
int attributeSize = attributeView.size(attributeName);
ByteBuffer attributeBuffer = ByteBuffer.allocate(attributeSize);
attributeView.read(attributeName, attributeBuffer);
attributeBuffer.flip();
String attributeValue = new String(attributeBuffer.array(), 0, attributeSize);
attrMap.put(attributeName, attributeValue);
}
return Collections.unmodifiableMap(attrMap);
} catch (UnsupportedOperationException e) {
LOG.warn("UFS does not support getting FileAttributeView with Java.");
throw new UnimplementedRuntimeException(e, ErrorType.External);
} catch (ClassCastException | IllegalArgumentException e) {
String msg = "UFS does not support getting FileAttributeView with Java.";
LOG.warn(msg);
throw new InvalidArgumentRuntimeException(msg, e);
} catch (SecurityException e) {
LOG.warn("UFS does not support getting FileAttributeView with Java.");
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
LOG.warn("UFS does not support getting FileAttributeView with Java.");
throw new UnknownRuntimeException(e);
}
}
/**
* Changes local file's permission to be "rwxrwxrwx".
*
* @param filePath that will change permission
*/
public static void changeLocalFileToFullPermission(String filePath) {
changeLocalFilePermission(filePath, "rwxrwxrwx");
}
/**
* Gets local file's permission mode.
*
* @param filePath the file path
* @return the file mode in short, e.g. 0777
*/
public static short getLocalFileMode(String filePath) throws IOException {
Set permission =
Files.readAttributes(Paths.get(filePath), PosixFileAttributes.class).permissions();
return translatePosixPermissionToMode(permission);
}
/**
* Translate posix file permissions to short mode.
*
* @param permission posix file permission
* @return mode for file
*/
public static short translatePosixPermissionToMode(Set permission) {
int mode = 0;
for (PosixFilePermission action : PosixFilePermission.values()) {
mode = mode << 1;
mode += permission.contains(action) ? 1 : 0;
}
return (short) mode;
}
/**
* Translate mode to posix file permissions.
*
* @param mode the file mode
* @return posix file permissions
*/
public static Set translateModeToPosixPermissions(int mode) {
Set perms = new HashSet<>();
// add owners permission
Preconditions.checkArgument(mode >= 0, "Mode can not be a negative value");
if ((mode & 0400) != 0) {
perms.add(PosixFilePermission.OWNER_READ);
}
if ((mode & 0200) != 0) {
perms.add(PosixFilePermission.OWNER_WRITE);
}
if ((mode & 0100) != 0) {
perms.add(PosixFilePermission.OWNER_EXECUTE);
}
// add group permissions
if ((mode & 0040) != 0) {
perms.add(PosixFilePermission.GROUP_READ);
}
if ((mode & 0020) != 0) {
perms.add(PosixFilePermission.GROUP_WRITE);
}
if ((mode & 0010) != 0) {
perms.add(PosixFilePermission.GROUP_EXECUTE);
}
// add others permissions
if ((mode & 0004) != 0) {
perms.add(PosixFilePermission.OTHERS_READ);
}
if ((mode & 0002) != 0) {
perms.add(PosixFilePermission.OTHERS_WRITE);
}
if ((mode & 0001) != 0) {
perms.add(PosixFilePermission.OTHERS_EXECUTE);
}
return perms;
}
/**
* Changes the local file's user.
*
* @param path that will change owner
* @param user the new user
*/
public static void changeLocalFileUser(String path, String user) throws IOException {
UserPrincipalLookupService lookupService =
FileSystems.getDefault().getUserPrincipalLookupService();
PosixFileAttributeView view =
Files.getFileAttributeView(Paths.get(path), PosixFileAttributeView.class,
LinkOption.NOFOLLOW_LINKS);
UserPrincipal userPrincipal = lookupService.lookupPrincipalByName(user);
view.setOwner(userPrincipal);
}
/**
* Sticky bit can be set primarily on directories in UNIX / Linux.
*
* If the sticky bit of is enabled on a directory, only the owner and the root user can delete /
* rename the files or directories within that directory. No one else can delete other users data
* in this directory(Where sticky bit is set).
*
* This is a security measure to avoid deletion of folders and their content (sub-folders and
* files), though other users have full permissions.
*
* Setting the sticky bit of a file is a no-op.
*
* @param dir absolute dir path to set the sticky bit
*/
public static void setLocalDirStickyBit(String dir) {
try {
// Support for sticky bit is platform specific. Check if the path starts with "/" and if so,
// assume that the host supports the chmod command.
if (dir.startsWith(AlluxioURI.SEPARATOR)) {
// TODO(peis): This is very slow. Consider removing this.
Runtime.getRuntime().exec("chmod +t " + dir);
}
} catch (IOException e) {
LOG.info("Can not set the sticky bit of the directory: {}", dir, e);
}
}
/**
* Creates the local block path and all the parent directories. Also, sets the appropriate
* permissions.
*
* @param path the path of the block
* @param workerDataFolderPermissions The permissions to set on the worker's data folder
*/
@VisibleForTesting
public static void createBlockPath(String path, String workerDataFolderPermissions) {
try {
String parent = PathUtils.getParent(path);
createStorageDirPath(parent, workerDataFolderPermissions);
} catch (InvalidPathException e) {
throw new InternalRuntimeException(
"Failed to create block path, get parent path of " + path + "failed", e);
}
}
/**
* Moves file from one place to another, can across storage devices (e.g., from memory to SSD)
* when {@link File#renameTo} may not work.
*
* @param srcPath pathname string of source file
* @param dstPath pathname string of destination file
*/
public static void move(String srcPath, String dstPath) throws IOException {
Files.move(Paths.get(srcPath), Paths.get(dstPath), StandardCopyOption.REPLACE_EXISTING);
}
/**
* Deletes the file or directory.
*
* @param path pathname string of file or directory
*/
public static void delete(String path) {
try {
Files.delete(Paths.get(path));
} catch (java.nio.file.InvalidPathException e) {
throw new InvalidArgumentRuntimeException(e);
} catch (NoSuchFileException e) {
throw new NotFoundRuntimeException(e);
} catch (DirectoryNotEmptyException e) {
throw new FailedPreconditionRuntimeException(e);
} catch (SecurityException e) {
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
throw new UnknownRuntimeException(e);
}
}
/**
* Deletes the file or directory, if it exists.
*
* @param path pathname string of file or directory
*/
public static void deleteIfExists(String path) {
try {
Files.deleteIfExists(Paths.get(path));
} catch (java.nio.file.InvalidPathException e) {
throw new InvalidArgumentRuntimeException(e);
} catch (DirectoryNotEmptyException e) {
throw new FailedPreconditionRuntimeException(e);
} catch (SecurityException e) {
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
throw new UnknownRuntimeException(e);
}
}
/**
* Deletes a file or a directory, recursively if it is a directory.
*
* If the path does not exist, nothing happens.
*
* @param path pathname to be deleted
*/
public static void deletePathRecursively(String path) throws IOException {
if (!exists(path)) {
return;
}
Path root = Paths.get(path);
Files.walkFileTree(root, new SimpleFileVisitor() {
@Override
public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
Files.delete(file);
return FileVisitResult.CONTINUE;
}
@Override
public FileVisitResult postVisitDirectory(Path dir, IOException e) throws IOException {
if (e == null) {
Files.delete(dir);
return FileVisitResult.CONTINUE;
} else {
throw e;
}
}
});
}
/**
* Creates the storage directory path, including any necessary but nonexistent parent directories.
* If the directory already exists, do nothing.
*
* Also, appropriate directory permissions (w/ StickyBit) are set.
*
* @param path storage directory path to create
* @param workerDataFolderPermissions the permissions to set for the worker's data folder
* @return true if the directory is created and false if the directory already exists
*/
public static boolean createStorageDirPath(String path, String workerDataFolderPermissions) {
if (Files.exists(Paths.get(path))) {
return false;
}
Path storagePath;
try {
storagePath = Files.createDirectories(Paths.get(path));
} catch (UnsupportedOperationException e) {
throw new UnimplementedRuntimeException(e, ErrorType.External);
} catch (FileAlreadyExistsException e) {
throw new AlreadyExistsRuntimeException(e);
} catch (SecurityException e) {
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
throw new UnknownRuntimeException(e);
}
String absolutePath = storagePath.toAbsolutePath().toString();
changeLocalFilePermission(absolutePath, workerDataFolderPermissions);
setLocalDirStickyBit(absolutePath);
return true;
}
/**
* Creates an empty file and its intermediate directories if necessary.
*
* @param filePath pathname string of the file to create
*/
public static void createFile(String filePath) {
Path storagePath = Paths.get(filePath);
Path parent = storagePath.getParent();
try {
if (parent != null) {
Files.createDirectories(parent);
}
Files.createFile(storagePath);
} catch (UnsupportedOperationException e) {
throw new UnimplementedRuntimeException(e, ErrorType.External);
} catch (ClassCastException e) {
throw new InvalidArgumentRuntimeException(e);
} catch (SecurityException e) {
throw new PermissionDeniedRuntimeException(e);
} catch (IOException e) {
throw new UnknownRuntimeException(e);
}
}
/**
* Creates an empty directory and its intermediate directories if necessary.
*
* @param path path of the directory to create
*/
public static void createDir(String path) throws IOException {
Files.createDirectories(Paths.get(path));
}
/**
* Checks if a path exists.
*
* @param path the given path
* @return true if path exists, false otherwise
*/
public static boolean exists(String path) {
return Files.exists(Paths.get(path));
}
/**
* Checks if a storage directory path is accessible.
*
* @param path the given path
* @return true if path exists, false otherwise
*/
public static boolean isStorageDirAccessible(String path) {
Path filePath = Paths.get(path);
return Files.exists(filePath)
&& Files.isReadable(filePath)
&& Files.isWritable(filePath)
&& Files.isExecutable(filePath);
}
private FileUtils() {} // prevent instantiation
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy