alluxio.underfs.ObjectUnderFileSystem 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.underfs;
import alluxio.AlluxioURI;
import alluxio.collections.Pair;
import alluxio.conf.AlluxioConfiguration;
import alluxio.conf.PropertyKey;
import alluxio.exception.ExceptionMessage;
import alluxio.exception.runtime.AlluxioRuntimeException;
import alluxio.retry.CountingRetry;
import alluxio.retry.ExponentialBackoffRetry;
import alluxio.retry.RetryPolicy;
import alluxio.underfs.options.CreateOptions;
import alluxio.underfs.options.DeleteOptions;
import alluxio.underfs.options.FileLocationOptions;
import alluxio.underfs.options.GetStatusOptions;
import alluxio.underfs.options.ListOptions;
import alluxio.underfs.options.MkdirsOptions;
import alluxio.underfs.options.OpenOptions;
import alluxio.util.CommonUtils;
import alluxio.util.executor.ExecutorServiceFactories;
import alluxio.util.io.PathUtils;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.Iterators;
import io.grpc.Status;
import org.apache.http.conn.ConnectTimeoutException;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.EOFException;
import java.io.FileNotFoundException;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.net.SocketException;
import java.net.UnknownHostException;
import java.text.MessageFormat;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.Comparator;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.List;
import java.util.Map;
import java.util.NavigableMap;
import java.util.TreeMap;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
import java.util.function.Supplier;
import javax.annotation.Nullable;
import javax.annotation.concurrent.ThreadSafe;
/**
* An object based abstract {@link UnderFileSystem}. Object Stores implementing the
* {@link UnderFileSystem} interface should derive from this class.
*/
@ThreadSafe
public abstract class ObjectUnderFileSystem extends BaseUnderFileSystem {
private static final Logger LOG = LoggerFactory.getLogger(ObjectUnderFileSystem.class);
/** Default maximum length for a single listing query. */
private static final int DEFAULT_MAX_LISTING_CHUNK_LENGTH = 1000;
/** Value used to indicate nested structure. */
protected static final char PATH_SEPARATOR_CHAR = '/';
/**
* Value used to indicate nested structure. This is a string representation of
* {@link ObjectUnderFileSystem#PATH_SEPARATOR_CHAR}.
*/
protected static final String PATH_SEPARATOR = String.valueOf(PATH_SEPARATOR_CHAR);
/** Executor service used for parallel UFS operations such as bulk deletes. */
protected ExecutorService mExecutorService;
/** The root key of an object fs. */
protected final Supplier mRootKeySupplier =
CommonUtils.memoize(this::getRootKey);
private final boolean mBreadcrumbsEnabled;
/**
* Constructs an {@link ObjectUnderFileSystem}.
*
* @param uri the {@link AlluxioURI} used to create this ufs
* @param ufsConf UFS configuration
*/
protected ObjectUnderFileSystem(AlluxioURI uri, UnderFileSystemConfiguration ufsConf) {
super(uri, ufsConf);
int numThreads = mUfsConf.getInt(PropertyKey.UNDERFS_OBJECT_STORE_SERVICE_THREADS);
mExecutorService = ExecutorServiceFactories.fixedThreadPool(
"alluxio-underfs-object-service-worker", numThreads).create();
mBreadcrumbsEnabled = mUfsConf.getBoolean(PropertyKey.UNDERFS_OBJECT_STORE_BREADCRUMBS_ENABLED);
}
/**
* Information about a single object in object UFS.
*/
protected static class ObjectStatus {
private static final long INVALID_CONTENT_LENGTH = -1L;
private final String mContentHash;
private final long mContentLength;
/** Last modified epoch time in ms, or null if it is not available. */
private final Long mLastModifiedTimeMs;
private final String mName;
public ObjectStatus(String name, String contentHash, long contentLength,
@Nullable Long lastModifiedTimeMs) {
mContentHash = contentHash == null ? UfsFileStatus.INVALID_CONTENT_HASH : contentHash;
mContentLength = contentLength;
mLastModifiedTimeMs = lastModifiedTimeMs;
mName = name;
}
public ObjectStatus(String name) {
mContentHash = UfsFileStatus.INVALID_CONTENT_HASH;
mContentLength = INVALID_CONTENT_LENGTH;
mLastModifiedTimeMs = null;
mName = name;
}
/**
* @return the hash of the content
*/
public String getContentHash() {
return mContentHash;
}
/**
* Gets the size of object contents in bytes.
*
* @return the content length in bytes
*/
public long getContentLength() {
return mContentLength;
}
/**
* Gets the last modified epoch time in ms.
*
* @return modification time in milliseconds
*/
@Nullable
public Long getLastModifiedTimeMs() {
return mLastModifiedTimeMs;
}
/**
* Gets the name of the object.
*
* @return object name
*/
public String getName() {
return mName;
}
}
/**
* A chunk of listing results.
*/
public interface ObjectListingChunk {
/**
* Returns objects in a pseudo-directory which may be a file or a directory.
*
* @return a list of object statuses
*/
ObjectStatus[] getObjectStatuses();
/**
* Use common prefixes to infer pseudo-directories in object store.
*
* @return a list of common prefixes
*/
String[] getCommonPrefixes();
/**
* Gets next chunk of object listings.
*
* @return null if listing did not find anything or is done, otherwise return new
* {@link ObjectListingChunk} for the next chunk
*/
@Nullable
ObjectListingChunk getNextChunk() throws IOException;
/**
* Gets if there is more chunks to fetch WITHOUT actually fetching the next chunk.
* @return true if there is, no if there isn't, NULL if it cannot tell
*/
default @Nullable Boolean hasNextChunk() {
throw new UnsupportedOperationException(
"HasNextChunk not implemented for " + getClass().getName());
}
}
/**
* Permissions in object UFS.
*/
public static class ObjectPermissions {
final String mOwner;
final String mGroup;
final short mMode;
/**
* Creates a new ObjectPermissions.
*
* @param owner the owner of the object
* @param group the group of the object
* @param mode the representation of the permission bits
*/
public ObjectPermissions(String owner, String group, short mode) {
mOwner = owner;
mGroup = group;
mMode = mode;
}
/**
* @return the name of the owner
*/
public String getOwner() {
return mOwner;
}
/**
* @return the name of the group
*/
public String getGroup() {
return mGroup;
}
/**
* @return the short representation of the permission bits
*/
public short getMode() {
return mMode;
}
}
/**
* Operations added to this buffer are performed concurrently.
* Note that {@link #getResult()} method blocks {@link #add(Object)} method.
*
* @param input type for operation
*/
@ThreadSafe
protected abstract class OperationBuffer {
/** A list of the successful operations for each batch. */
private final ArrayList>> mBatchesResult;
/** Buffer for a batch of inputs. */
private final List mCurrentBatchBuffer;
/** Total number of inputs to be operated on across batches. */
protected int mEntriesAdded;
/**
* Construct a new {@link OperationBuffer} instance.
*/
protected OperationBuffer() {
mBatchesResult = new ArrayList<>();
mCurrentBatchBuffer = new ArrayList<>();
mEntriesAdded = 0;
}
/**
* Gets the batch size.
*
* @return a positive integer denoting the batch size
*/
protected abstract int getBatchSize();
/**
* Operate on a list of input type {@link T}.
*
* @param paths the list of input type {@link T} to operate on
* @return list of inputs for successful operations
*/
protected abstract List operate(List paths) throws IOException;
/**
* Adds a new input to be operated on.
*
* @param input the input to operate on
* @throws IOException if a non-Alluxio error occurs
*/
public synchronized void add(T input) throws IOException {
if (mCurrentBatchBuffer.size() == getBatchSize()) {
// Batch is full
submitBatch();
}
mCurrentBatchBuffer.add(input);
mEntriesAdded++;
}
/**
* Gets the combined result from all batches.
*
* @return a list of inputs for successful operations
* @throws IOException if a non-Alluxio error occurs
*/
public synchronized List getResult() throws IOException {
submitBatch();
List result = new ArrayList<>();
for (Future> list : mBatchesResult) {
try {
result.addAll(list.get());
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
// If operation was interrupted do not add to successfully deleted list
LOG.warn(
"{}: Interrupted while waiting for the result of batch operation. UFS and Alluxio "
+ "state may be inconsistent. Error: {}",
getClass().getName(), e.getMessage());
} catch (ExecutionException e) {
// If operation failed to execute do not add to successfully deleted list
LOG.warn(
"{}: A batch operation failed. UFS and Alluxio state may be inconsistent. Error: {}",
getClass().getName(), e.getMessage());
}
}
return result;
}
/**
* Process the current batch asynchronously.
*/
private void submitBatch() throws IOException {
if (mCurrentBatchBuffer.size() != 0) {
List batch = new ArrayList<>(mCurrentBatchBuffer);
mCurrentBatchBuffer.clear();
mBatchesResult.add(mExecutorService.submit(() -> {
try {
return operate(batch);
} catch (IOException e) {
LOG.error("A batch operation failed. ", e);
// Do not append to success list
return Collections.emptyList();
}
}));
}
}
}
@Override
public void cleanup() throws IOException {
}
@Override
public void close() throws IOException {
}
@Override
public void connectFromMaster(String hostname) {
}
@Override
public void connectFromWorker(String hostname) {
}
@Override
public OutputStream create(String path, CreateOptions options) throws IOException {
if (options.getCreateParent()
&& !mUfsConf.getBoolean(PropertyKey.UNDERFS_OBJECT_STORE_SKIP_PARENT_DIRECTORY_CREATION)
&& !mkdirs(getParentPath(path))) {
throw new IOException(ExceptionMessage.PARENT_CREATION_FAILED.getMessage(path));
}
return createObject(stripPrefixIfPresent(path));
}
@Override
public OutputStream createNonexistingFile(String path) throws IOException {
return retryOnException(() -> create(path), () -> "create file " + path);
}
@Override
public OutputStream createNonexistingFile(String path, CreateOptions options) throws IOException {
return retryOnException(() -> create(path, options),
() -> "create file " + path + " with options " + options);
}
@Override
public boolean deleteFile(String path) throws IOException {
return deleteObject(stripPrefixIfPresent(path));
}
@Override
public boolean deleteExistingFile(String path) throws IOException {
return retryOnFalse(() -> deleteFile(path), () -> "delete existing file " + path);
}
@Override
public boolean deleteDirectory(String path, DeleteOptions options) throws IOException {
if (!options.isRecursive()) {
UfsStatus[] children = listInternal(path, ListOptions.defaults());
if (children == null) {
LOG.error("Unable to delete path because {} is not a directory ", path);
return false;
}
if (children.length != 0) {
LOG.error("Unable to delete {} because it is a non empty directory. Specify "
+ "recursive as true in order to delete non empty directories.", path);
return false;
}
// Delete the directory itself
return deleteObject(stripPrefixIfPresent(convertToFolderName(path)));
}
// Delete children
DeleteBuffer deleteBuffer = new DeleteBuffer();
UfsStatus[] pathsToDelete = listInternal(path, ListOptions.defaults().setRecursive(true));
if (pathsToDelete == null) {
LOG.warn("Unable to delete {} because listInternal returns null", path);
return false;
}
Arrays.sort(pathsToDelete, Comparator.comparing(UfsStatus::getName).reversed());
for (UfsStatus pathToDelete : pathsToDelete) {
String pathKey = stripPrefixIfPresent(PathUtils.concatPath(path, pathToDelete.getName()));
if (pathToDelete.isDirectory()) {
deleteBuffer.add(convertToFolderName(pathKey));
} else {
deleteBuffer.add(pathKey);
}
}
deleteBuffer.add(stripPrefixIfPresent(convertToFolderName(path)));
int filesDeleted = deleteBuffer.getResult().size();
if (filesDeleted != deleteBuffer.mEntriesAdded) {
LOG.warn("Failed to delete directory, successfully deleted {} files out of {}.",
filesDeleted, deleteBuffer.mEntriesAdded);
return false;
}
return true;
}
@Override
public boolean deleteExistingDirectory(String path) throws IOException {
return retryOnFalse(() -> deleteDirectory(path), () -> "delete directory " + path);
}
@Override
public boolean deleteExistingDirectory(String path, DeleteOptions options) throws IOException {
return retryOnFalse(() -> deleteDirectory(path, options),
() -> "delete directory " + path + " with options " + options);
}
/**
* Object keys added to a {@link DeleteBuffer} will be deleted in batches.
*/
@ThreadSafe
protected class DeleteBuffer extends OperationBuffer {
/**
* Construct a new {@link DeleteBuffer} instance.
*/
public DeleteBuffer() {}
@Override
protected int getBatchSize() {
// Delete batch size is same as listing length
return getListingChunkLength(mUfsConf);
}
@Override
protected List operate(List paths) throws IOException {
return deleteObjects(paths);
}
}
/**
* Gets the block size in bytes. This method defaults to the default user block size in Alluxio.
*
* @param path the file name
* @return the default Alluxio user block size
*/
@Override
public long getBlockSizeByte(String path) throws IOException {
return mUfsConf.getBytes(PropertyKey.USER_BLOCK_SIZE_BYTES_DEFAULT);
}
@Override
public UfsDirectoryStatus getDirectoryStatus(String path) throws IOException {
if (isDirectory(path)) {
ObjectPermissions permissions = getPermissions();
return new UfsDirectoryStatus(path, permissions.getOwner(), permissions.getGroup(),
permissions.getMode());
}
LOG.debug("Error fetching directory status, assuming directory {} does not exist", path);
throw new FileNotFoundException("Failed to fetch directory status " + path);
}
@Override
public UfsDirectoryStatus getExistingDirectoryStatus(String path) throws IOException {
return retryOnException(() -> getDirectoryStatus(path),
() -> "get status of directory " + path);
}
// Not supported
@Override
@Nullable
public List getFileLocations(String path) throws IOException {
LOG.debug("getFileLocations is not supported when using default ObjectUnderFileSystem.");
return null;
}
// Not supported
@Override
@Nullable
public List getFileLocations(String path, FileLocationOptions options)
throws IOException {
LOG.debug("getFileLocations is not supported when using default ObjectUnderFileSystem.");
return null;
}
// This call is currently only used for the web ui, where a negative value implies unknown.
@Override
public long getSpace(String path, SpaceType type) throws IOException {
return -1;
}
@Override
public UfsFileStatus getFileStatus(String path, GetStatusOptions options) throws IOException {
ObjectStatus details = getObjectStatus(stripPrefixIfPresent(path));
if (details != null) {
ObjectPermissions permissions = getPermissions();
return new UfsFileStatus(path, details.getContentHash(), details.getContentLength(),
details.getLastModifiedTimeMs(), permissions.getOwner(), permissions.getGroup(),
permissions.getMode(), mUfsConf.getBytes(PropertyKey.USER_BLOCK_SIZE_BYTES_DEFAULT));
} else {
LOG.debug("Error fetching file status, assuming file {} does not exist", path);
throw new FileNotFoundException("Failed to fetch file status " + path);
}
}
@Override
public UfsFileStatus getExistingFileStatus(String path) throws IOException {
return retryOnException(() -> getFileStatus(path), () -> "get status of file " + path);
}
@Override
public UfsStatus getStatus(String path, GetStatusOptions options) throws IOException {
if (isRoot(path)) {
return getDirectoryStatus(path);
}
ObjectStatus details = getObjectStatus(stripPrefixIfPresent(path));
if (details != null) {
ObjectPermissions permissions = getPermissions();
return new UfsFileStatus(path, details.getContentHash(), details.getContentLength(),
details.getLastModifiedTimeMs(), permissions.getOwner(), permissions.getGroup(),
permissions.getMode(), mUfsConf.getBytes(PropertyKey.USER_BLOCK_SIZE_BYTES_DEFAULT));
}
return getDirectoryStatus(path);
}
@Override
public UfsStatus getExistingStatus(String path) throws IOException {
return retryOnException(() -> getStatus(path),
() -> "get status of " + path);
}
@Override
public boolean isDirectory(String path) throws IOException {
// Root is always a folder
if (isRoot(path)) {
return true;
}
String keyAsFolder = convertToFolderName(stripPrefixIfPresent(path));
if (getObjectStatus(keyAsFolder) != null) {
return true;
}
return getObjectListingChunkForPath(path, true) != null;
}
@Override
public boolean isExistingDirectory(String path) throws IOException {
return retryOnException(() -> isDirectory(path),
() -> "check if " + path + " is a directory");
}
@Override
public void setAttribute(String path, String name, byte[] value) throws IOException {
path = stripPrefixIfPresent(path);
if (isDirectory(path)) {
setObjectTagging(convertToFolderName(path), name, new String(value));
} else {
setObjectTagging(path, name, new String(value));
}
}
@Override
public Map getAttributes(String path) throws IOException {
path = stripPrefixIfPresent(path);
try {
if (isDirectory(path)) {
return getObjectTags(convertToFolderName(path));
} else {
return getObjectTags(path);
}
} catch (AlluxioRuntimeException e) {
if (e.getStatus().getCode() == Status.Code.NOT_FOUND) {
return null;
}
throw e;
}
}
/**
* Sets a tag to an object in the ObjectUnderFilesystem.
* @param path file path
* @param name the name of the tag
* @param value the value of the tag
* @throws IOException
*/
protected abstract void setObjectTagging(String path, String name, String value)
throws IOException;
/**
* Gets the all tags of an object in ObjectUnderFilesystem.
* @param path file path
* @return the map contains all tags
* @throws IOException
*/
protected abstract Map getObjectTags(String path) throws IOException;
@Override
public boolean isFile(String path) throws IOException {
// Directly try to get the file metadata, if we fail it either is a folder or does not exist
return !isRoot(path) && (getObjectStatus(stripPrefixIfPresent(path)) != null);
}
@Override
public boolean isObjectStorage() {
return true;
}
@Override
public UfsStatus[] listStatus(String path) throws IOException {
return listInternal(path, ListOptions.defaults());
}
@Override
public UfsStatus[] listStatus(String path, ListOptions options)
throws IOException {
return listInternal(path, options);
}
@Nullable
@Override
public Iterator listStatusIterable(
String path, ListOptions options, String startAfter, int batchSize) throws IOException {
final ObjectListingChunk chunk =
getObjectListingChunkForPath(path, options.isRecursive(), startAfter, batchSize);
if (chunk == null) {
String keyAsFolder = convertToFolderName(stripPrefixIfPresent(path));
if (getObjectStatus(keyAsFolder) != null) {
// Path is an empty directory
return Collections.emptyIterator();
}
return null;
}
return new UfsStatusIterator(path, options.isRecursive(), chunk);
}
@Override
public boolean mkdirs(String path, MkdirsOptions options) throws IOException {
if (path == null) {
return false;
}
if (isDirectory(path)) {
return true;
}
if (isFile(path)) {
LOG.error("Cannot create directory {} because it is already a file.", path);
return false;
}
if (!options.getCreateParent()) {
if (parentExists(path)) {
// Parent directory exists
return mkdirsInternal(path);
} else {
LOG.error("Cannot create directory {} because parent does not exist", path);
return false;
}
}
// Parent directories should be created
if (parentExists(path)) {
// Parent directory exists
return mkdirsInternal(path);
} else {
String parentKey = getParentPath(path);
// Recursively make the parent folders
return mkdirs(parentKey) && mkdirsInternal(path);
}
}
@Override
public InputStream open(String path, OpenOptions options) throws IOException {
return openObject(stripPrefixIfPresent(path), options, getRetryOncePolicy());
}
@Override
public InputStream openExistingFile(String path) throws IOException {
return openExistingFile(path, OpenOptions.defaults());
}
@Override
public InputStream openExistingFile(String path, OpenOptions options) throws IOException {
return openObject(stripPrefixIfPresent(path), options, getRetryPolicy());
}
@Override
public boolean renameDirectory(String src, String dst) throws IOException {
if (exists(dst)) {
LOG.error("Unable to rename {} to {} because destination already exists.", src, dst);
return false;
}
// Use a global delete buffer, in order to merge delete object requests
DeleteBuffer deleteBuffer = new DeleteBuffer();
boolean result = renameDirectoryInternal(src, dst, deleteBuffer);
int fileDeleted = deleteBuffer.getResult().size();
if (fileDeleted != deleteBuffer.mEntriesAdded) {
LOG.warn("Failed to rename directory, successfully deleted {} files out of {}.",
fileDeleted, deleteBuffer.mEntriesAdded);
return false;
}
return result;
}
private boolean renameDirectoryInternal(String src, String dst, DeleteBuffer deleteBuffer)
throws IOException {
UfsStatus[] children = listInternal(src, ListOptions.defaults());
if (children == null) {
LOG.error("Failed to list directory {}, aborting rename.", src);
return false;
}
// Source exists and is a directory, and destination does not exist
// Rename the source folder first
String srcKey = stripPrefixIfPresent(convertToFolderName(src));
if (!copyObject(srcKey, stripPrefixIfPresent(convertToFolderName(dst)))) {
return false;
}
deleteBuffer.add(srcKey);
// Rename each child in the src folder to destination/child
// a. Since renames are a copy operation, files are added to a buffer and processed concurrently
// b. Pseudo-directories are metadata only operations are not added to the buffer
RenameBuffer buffer = new RenameBuffer(deleteBuffer);
for (UfsStatus child : children) {
String childSrcPath = PathUtils.concatPath(src, child.getName());
String childDstPath = PathUtils.concatPath(dst, child.getName());
if (child.isDirectory()) {
// Recursive call
if (!renameDirectoryInternal(childSrcPath, childDstPath, deleteBuffer)) {
LOG.error("Failed to rename path {} to {}, aborting rename.", childSrcPath, childDstPath);
return false;
}
} else {
buffer.add(new Pair<>(childSrcPath, childDstPath));
}
}
// Get result of parallel file renames
int filesRenamed = buffer.getResult().size();
if (filesRenamed != buffer.mEntriesAdded) {
LOG.warn("Failed to rename directory, successfully renamed {} files out of {}.",
filesRenamed, buffer.mEntriesAdded);
return false;
}
return true;
}
@Override
public boolean renameRenamableDirectory(String src, String dst) throws IOException {
return retryOnFalse(() -> renameDirectory(src, dst),
() -> "rename directory from " + src + " to " + dst);
}
/**
* File paths added to a {@link RenameBuffer} will be renamed concurrently.
*/
@ThreadSafe
protected class RenameBuffer extends OperationBuffer> {
private final DeleteBuffer mDeleteBuffer;
/**
* Construct a new {@link RenameBuffer} instance.
*
* @param deleteBuffer delete object buffer
*/
public RenameBuffer(DeleteBuffer deleteBuffer) {
mDeleteBuffer = deleteBuffer;
}
@Override
protected int getBatchSize() {
return 1;
}
@Override
protected List> operate(List> paths)
throws IOException {
List> succeeded = new ArrayList<>();
for (Pair pathPair : paths) {
String src = stripPrefixIfPresent(pathPair.getFirst());
String dst = stripPrefixIfPresent(pathPair.getSecond());
if (copyObject(src, dst)) {
mDeleteBuffer.add(src);
succeeded.add(pathPair);
}
}
return succeeded;
}
}
@Override
public boolean renameFile(String src, String dst) throws IOException {
if (!isFile(src)) {
LOG.error("Unable to rename {} to {} because source does not exist or is a directory.",
src, dst);
return false;
}
if (exists(dst)) {
LOG.error("Unable to rename {} to {} because destination already exists.", src, dst);
return false;
}
// Source is a file and Destination does not exist
return copyObject(stripPrefixIfPresent(src), stripPrefixIfPresent(dst))
&& deleteObject(stripPrefixIfPresent(src));
}
@Override
public boolean renameRenamableFile(String src, String dst) throws IOException {
return retryOnFalse(() -> renameFile(src, dst),
() -> "rename file from " + src + " to " + dst);
}
@Override
public boolean supportsFlush() throws IOException {
return false;
}
/**
* Creates a zero-byte object used to encode a directory.
*
* @param key the key to create
* @return true if the operation was successful
*/
@VisibleForTesting
public abstract boolean createEmptyObject(String key);
/**
* Creates an {@link OutputStream} for object uploads.
*
* @param key ufs key including scheme and bucket
* @return new OutputStream
*/
protected abstract OutputStream createObject(String key) throws IOException;
/**
* Appends the directory suffix to the key.
*
* @param key the key to convert
* @return key as a directory path
*/
protected String convertToFolderName(String key) {
// Strips the slash if it is the end of the key string. This is because the slash at
// the end of the string is not part of the Object key.
key = CommonUtils.stripSuffixIfPresent(key, PATH_SEPARATOR);
return key + getFolderSuffix();
}
/**
* Copies an object to another key.
*
* @param src the source key to copy
* @param dst the destination key to copy to
* @return true if the operation was successful, false otherwise
*/
protected abstract boolean copyObject(String src, String dst) throws IOException;
/**
* Internal function to delete a key.
*
* @param key the key to delete
* @return true if successful, false if an exception is thrown
*/
protected abstract boolean deleteObject(String key) throws IOException;
/**
* Internal function to delete a list of keys.
*
* @param keys the list of keys to delete
* @return list of successfully deleted keys
*/
protected List deleteObjects(List keys) throws IOException {
List result = new ArrayList<>();
for (String key : keys) {
boolean status = deleteObject(key);
// If key is a directory, it is possible that it was not created through Alluxio and no
// zero-byte breadcrumb exists
if (status || key.endsWith(getFolderSuffix())) {
result.add(key);
}
}
return result;
}
/**
* Permissions for the mounted bucket.
*
* @return permissions
*/
protected abstract ObjectPermissions getPermissions();
/**
* Maximum number of items in a single listing chunk supported by the under store.
*
* @return the maximum length for a single listing query
*/
protected int getListingChunkLengthMax() {
return DEFAULT_MAX_LISTING_CHUNK_LENGTH;
}
/**
* The number of items to query in a single listing chunk.
*
* @return length of each list request
*/
protected int getListingChunkLength(AlluxioConfiguration conf) {
return Math.min(conf.getInt(PropertyKey.UNDERFS_LISTING_LENGTH), getListingChunkLengthMax());
}
/**
* Get metadata information about object. Implementations should process the key as is, which
* may be a file or a directory key.
*
* @param key ufs key to get metadata for
* @return {@link ObjectStatus} if key exists and successful, otherwise null
*/
@Nullable
protected abstract ObjectStatus getObjectStatus(String key) throws IOException;
/**
* Get parent path.
*
* @param path ufs path including scheme and bucket
* @return the parent path, or null if the parent does not exist
*/
@Nullable
protected String getParentPath(String path) {
// Root does not have a parent.
if (isRoot(path)) {
return null;
}
int separatorIndex = path.lastIndexOf(PATH_SEPARATOR);
if (separatorIndex < 0) {
return null;
}
return path.substring(0, separatorIndex);
}
/**
* Checks if the path is the root. This method supports full path (e.g. s3://bucket_name/dir)
* and stripped path (e.g. /dir).
*
* @param path ufs path including scheme and bucket
* @return true if the path is the root, false otherwise
*/
protected boolean isRoot(String path) {
String normalizePath = PathUtils.normalizePath(path, PATH_SEPARATOR);
return normalizePath.equals(PATH_SEPARATOR)
|| normalizePath.equals(PathUtils.normalizePath(mRootKeySupplier.get(), PATH_SEPARATOR));
}
/**
* Gets the child name based on the parent name.
*
* @param child the key of the child
* @param parent the key of the parent
* @return the child key with the parent prefix removed
*/
protected String getChildName(String child, String parent) throws IOException {
if (child.startsWith(parent)) {
return child.substring(parent.length());
}
throw new IOException(
MessageFormat.format("Parent path \"{0}\" is not a prefix of child {1}.", parent, child));
}
/**
* Get suffix used to encode a directory.
*
* @return folder suffix
*/
protected abstract String getFolderSuffix();
/**
* Gets a (partial) object listing result for the given key.
*
* @param key pseudo-directory key excluding header and bucket
* @param recursive whether to request immediate children only, or all descendants
* @param startAfter indicates where the listing starts
* @param batchSize the batch size of each chunk
* @return chunked object listing, or null if key is not found
*/
@Nullable
protected ObjectListingChunk getObjectListingChunk(
String key, boolean recursive, String startAfter, int batchSize) throws IOException {
// Some UFS haven't implemented getObjectListingChunk(dir, recursive, startAfter, batchSize)
// so falling back to the one with less param if startAfter and batchSize is unset.
if (startAfter == null && batchSize == 0) {
return getObjectListingChunk(key, recursive);
}
throw new UnsupportedOperationException("Operation not supported");
}
/**
* Gets a (partial) object listing result for the given key.
*
* @param key pseudo-directory key excluding header and bucket
* @param recursive whether to request immediate children only, or all descendants
* @return chunked object listing, or null if key is not found
*/
@Nullable
protected abstract ObjectListingChunk getObjectListingChunk(String key, boolean recursive)
throws IOException;
protected ObjectListingChunk getObjectListingChunkForPath(String path, boolean recursive)
throws IOException {
return getObjectListingChunkForPath(path, recursive, null, 0);
}
/**
* Gets a (partial) object listing for the given path.
*
* @param path of pseudo-directory
* @param recursive whether to request immediate children only, or all descendants
* @return chunked object listing, or null if the path does not exist as a pseudo-directory
*/
@Nullable
protected ObjectListingChunk getObjectListingChunkForPath(
String path, boolean recursive, String startAfter, int batchSize) throws IOException {
// Check if anything begins with /
String dir = stripPrefixIfPresent(path);
ObjectListingChunk objs = getObjectListingChunk(dir, recursive, startAfter, batchSize);
// If there are, this is a folder and we can create the necessary metadata
if (objs != null
&& ((objs.getObjectStatuses() != null && objs.getObjectStatuses().length > 0)
|| (objs.getCommonPrefixes() != null && objs.getCommonPrefixes().length > 0))) {
// Do not recreate the breadcrumb if it already exists
String folderName = convertToFolderName(dir);
if (!mUfsConf.isReadOnly() && mBreadcrumbsEnabled && !isRoot(dir)
&& Arrays.stream(objs.getObjectStatuses()).noneMatch(
x -> x.mContentLength == 0 && x.getName().equals(folderName))) {
mkdirsInternal(dir);
}
return objs;
}
return null;
}
private void populateUfsStatus(
String keyPrefix, ObjectListingChunk chunk,
boolean isRecursive, Map ufsStatusMap) throws IOException {
// Directories in UFS can be possibly encoded in two different ways:
// (1) as file objects with FOLDER_SUFFIX for directories created through Alluxio or
// (2) as "common prefixes" of other files objects for directories not created through
// Alluxio
//
// Case (1) (and file objects) is accounted for by iterating over chunk.getObjects() while
// case (2) is accounted for by iterating over chunk.getCommonPrefixes().
//
// An example, with prefix="ufs" and delimiter="/" and LISTING_LENGTH=5
// - objects.key = ufs/, child =
// - objects.key = ufs/dir1, child = dir1
// - objects.key = ufs/file, child = file
// - commonPrefix = ufs/dir1/, child = dir1
// - commonPrefix = ufs/dir2/, child = dir2
// Handle case (1)
for (ObjectStatus status : chunk.getObjectStatuses()) {
// Remove parent portion of the key
String child = getChildName(status.getName(), keyPrefix);
if (child.isEmpty() || child.equals(getFolderSuffix())) {
// Removes results equal to the path
continue;
}
ObjectPermissions permissions = getPermissions();
if (child.endsWith(getFolderSuffix())) {
// Child is a directory
child = CommonUtils.stripSuffixIfPresent(child, getFolderSuffix());
ufsStatusMap.put(child, new UfsDirectoryStatus(child, permissions.getOwner(),
permissions.getGroup(), permissions.getMode()));
} else {
// Child is a file
ufsStatusMap.put(child,
new UfsFileStatus(child, status.getContentHash(), status.getContentLength(),
status.getLastModifiedTimeMs(), permissions.getOwner(), permissions.getGroup(),
permissions.getMode(),
mUfsConf.getBytes(PropertyKey.USER_BLOCK_SIZE_BYTES_DEFAULT)));
}
}
// Handle case (2)
String[] commonPrefixes;
if (isRecursive) {
// In case of a recursive listing infer pseudo-directories as the commonPrefixes returned
// from the object store is empty for an empty delimiter.
HashSet prefixes = new HashSet<>();
for (ObjectStatus objectStatus : chunk.getObjectStatuses()) {
String objectName = objectStatus.getName();
while (objectName.startsWith(keyPrefix) && objectName.contains(PATH_SEPARATOR)) {
objectName = objectName.substring(0, objectName.lastIndexOf(PATH_SEPARATOR));
if (!objectName.isEmpty()) {
// include the separator with the prefix, to conform to what object stores return
// as common prefixes.
prefixes.add(PathUtils.normalizePath(objectName, PATH_SEPARATOR));
}
}
}
commonPrefixes = prefixes.toArray(new String[0]);
} else {
commonPrefixes = chunk.getCommonPrefixes();
}
for (String commonPrefix : commonPrefixes) {
if (commonPrefix.startsWith(keyPrefix)) {
// Remove parent portion of the key
String child = getChildName(commonPrefix, keyPrefix);
// Remove any portion after the last path delimiter
int childNameIndex = child.lastIndexOf(PATH_SEPARATOR);
child = childNameIndex != -1 ? child.substring(0, childNameIndex) : child;
if (!child.isEmpty() && !ufsStatusMap.containsKey(child)) {
// If both a file and a directory existed with the same name, the path will be
// treated as a directory
ObjectPermissions permissions = getPermissions();
ufsStatusMap.put(child, new UfsDirectoryStatus(child, permissions.getOwner(),
permissions.getGroup(), permissions.getMode()));
}
}
}
}
/**
* Get full path of root in object store.
*
* @return full path including scheme and bucket
*/
protected abstract String getRootKey();
/**
* Lists the files in the given path, the paths will be their logical names and not contain the
* folder suffix. Note that, the list results are unsorted.
*
* @param path the key to list
* @param options for listing
* @return an array of the file and folder names in this directory
*/
@Nullable
protected UfsStatus[] listInternal(String path, ListOptions options) throws IOException {
ObjectListingChunk chunk = getObjectListingChunkForPath(path, options.isRecursive());
if (chunk == null) {
String keyAsFolder = convertToFolderName(stripPrefixIfPresent(path));
if (getObjectStatus(keyAsFolder) != null) {
// Path is an empty directory
return new UfsStatus[0];
}
return null;
}
String keyPrefix = PathUtils.normalizePath(stripPrefixIfPresent(path), PATH_SEPARATOR);
keyPrefix = keyPrefix.equals(PATH_SEPARATOR) ? "" : keyPrefix;
Map children = new HashMap<>();
while (chunk != null) {
populateUfsStatus(keyPrefix, chunk, options.isRecursive(), children);
chunk = chunk.getNextChunk();
}
UfsStatus[] ret = new UfsStatus[children.size()];
int pos = 0;
for (UfsStatus status : children.values()) {
ret[pos++] = status;
}
return ret;
}
/**
* The UFS status iterator that iterates the ufs statuses and fetches the chunk by lazy.
*/
public class UfsStatusIterator implements Iterator {
private ObjectListingChunk mChunk;
private final String mKeyPrefix;
private final boolean mIsRecursive;
private Iterator mIterator = null;
private String mLastKey = null;
/**
* Creates the iterator.
* @param path the path
* @param isRecursive if the listing is recursive
* @param firstChunk the first object listing chunk
*/
public UfsStatusIterator(String path, boolean isRecursive, ObjectListingChunk firstChunk)
throws IOException {
String keyPrefix = PathUtils.normalizePath(stripPrefixIfPresent(path), PATH_SEPARATOR);
keyPrefix = keyPrefix.equals(PATH_SEPARATOR) ? "" : keyPrefix;
mKeyPrefix = keyPrefix;
mIsRecursive = isRecursive;
mChunk = firstChunk;
updateIterator();
}
private void updateIterator() throws IOException {
NavigableMap ufsStatusMap = new TreeMap<>();
populateUfsStatus(mKeyPrefix, mChunk, mIsRecursive, ufsStatusMap);
if (mLastKey != null) {
ufsStatusMap = ufsStatusMap.tailMap(mLastKey, false);
}
mIterator = Iterators.transform(ufsStatusMap.entrySet().iterator(), Map.Entry::getValue);
mLastKey = ufsStatusMap.isEmpty() ? null : ufsStatusMap.lastKey();
}
@Override
public boolean hasNext() {
if (mChunk == null) {
return false;
}
if (mIterator.hasNext()) {
return true;
}
if (Boolean.FALSE.equals(mChunk.hasNextChunk())) {
return false;
}
try {
mChunk = mChunk.getNextChunk();
updateIterator();
return hasNext();
} catch (Exception e) {
throw new RuntimeException(e);
}
}
@Override
public UfsStatus next() {
return mIterator.next();
}
}
/**
* Creates a directory flagged file with the key and folder suffix.
*
* @param key the key to create a folder
* @return true if the operation was successful, false otherwise
*/
protected boolean mkdirsInternal(String key) {
return createEmptyObject(convertToFolderName(stripPrefixIfPresent(key)));
}
/**
* Internal function to open an input stream to an object.
*
* @param key the key to open
* @param options the open options
* @param retryPolicy the retry policy of the opened stream to solve eventual consistency issue
* @return an {@link InputStream} to read from key
*/
protected abstract InputStream openObject(String key, OpenOptions options,
RetryPolicy retryPolicy) throws IOException;
/**
* Treating the object store as a file system, checks if the parent directory exists.
*
* @param path the path to check
* @return true if the parent exists or if the path is root, false otherwise
*/
protected boolean parentExists(String path) throws IOException {
// Assume root always has a parent
if (isRoot(path)) {
return true;
}
String parentKey = getParentPath(path);
return parentKey != null && isDirectory(parentKey);
}
/**
* Strips the bucket prefix or the preceding path separator from the path if it is present. For
* example, for input path ufs://my-bucket-name/my-path/file, the output would be my-path/file. If
* path is an absolute path like /my-path/file, the output would be my-path/file. This method will
* leave keys without a prefix unaltered, ie. my-path/file returns my-path/file.
*
* @param path the path to strip
* @return the path without the bucket prefix
*/
@VisibleForTesting
public String stripPrefixIfPresent(String path) {
if (mRootKeySupplier.get().equals(path)) {
return "";
}
String stripedKey = CommonUtils.stripPrefixIfPresent(
path, PathUtils.normalizePath(mRootKeySupplier.get(), PATH_SEPARATOR));
if (!stripedKey.equals(path)) {
return stripedKey;
}
return CommonUtils.stripPrefixIfPresent(path, PATH_SEPARATOR);
}
/**
* Represents an object store operation.
*/
@VisibleForTesting
protected interface ObjectStoreOperation {
/**
* Applies this operation.
*
* @return the result of this operation
*/
T apply() throws IOException;
}
/**
* Filters exception that need to be retried.
* if exception need to be retried will return to continue the retry.
* else will throw exception to quit retry.
* @param e exception to be handled
* @throws IOException Exceptions that do not need to be tried again will be thrown directly
*/
private void handleRetriablException(IOException e) throws IOException {
if (e instanceof EOFException
|| e instanceof UnknownHostException
|| e instanceof ConnectTimeoutException) {
LOG.warn("retry policy meet exception, and will retry, e:", e);
return;
} else if (e instanceof SocketException) {
LOG.warn("retry policy meet socket exception, and will retry, e:", e);
return;
} else {
LOG.warn("retry policy meet exception, but no need to retry, e:", e);
throw e;
}
}
/**
* Retries the given object store operation when it throws exceptions
* to resolve eventual consistency issue.
*
* @param op the object store operation to retry
* @param description the description regarding the operation
* @return the operation result if operation succeed
*/
@VisibleForTesting
protected T retryOnException(ObjectStoreOperation op,
Supplier description) throws IOException {
RetryPolicy retryPolicy = getRetryPolicy();
IOException thrownException = null;
while (retryPolicy.attempt()) {
try {
return op.apply();
} catch (IOException e) {
LOG.debug("Attempt {} to {} failed with exception : {}", retryPolicy.getAttemptCount(),
description.get(), e.toString());
handleRetriablException(e);
thrownException = e;
}
}
throw thrownException;
}
/**
* Retries the given object store operation when it returns false
* to resolve eventual consistency issue.
*
* @param op the object store operation to retry
* @param description the description regarding the operation
* @return the operation result if operation returned true
*/
private boolean retryOnFalse(ObjectStoreOperation op,
Supplier description) throws IOException {
RetryPolicy retryPolicy = getRetryPolicy();
while (retryPolicy.attempt()) {
if (op.apply()) {
return true;
} else {
LOG.debug("Failed in attempt {} to {} ", retryPolicy.getAttemptCount(),
description.get());
}
}
return false;
}
/**
* @return the exponential backoff retry policy to use
*/
private RetryPolicy getRetryPolicy() {
return new ExponentialBackoffRetry(
(int) mUfsConf.getMs(PropertyKey.UNDERFS_EVENTUAL_CONSISTENCY_RETRY_BASE_SLEEP_MS),
(int) mUfsConf.getMs(PropertyKey.UNDERFS_EVENTUAL_CONSISTENCY_RETRY_MAX_SLEEP_MS),
mUfsConf.getInt(PropertyKey.UNDERFS_EVENTUAL_CONSISTENCY_RETRY_MAX_NUM));
}
/**
* @return the retry once policy to use
*/
private RetryPolicy getRetryOncePolicy() {
return new CountingRetry(1);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy