
org.jclouds.blobstore.internal.BaseBlobStore Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of jclouds-shaded Show documentation
Show all versions of jclouds-shaded Show documentation
Provides a shaded jclouds with relocated guava and guice
The newest version!
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You under the Apache License, Version 2.0
* (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package org.jclouds.blobstore.internal;
import static shaded.com.google.common.base.Preconditions.checkNotNull;
import static shaded.com.google.common.base.Preconditions.checkState;
import static org.jclouds.Constants.PROPERTY_USER_THREADS;
import static org.jclouds.blobstore.options.ListContainerOptions.Builder.recursive;
import static org.jclouds.util.Predicates2.retry;
import java.io.File;
import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.Date;
import java.util.Map;
import java.util.Set;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutorService;
import javax.inject.Named;
import org.jclouds.blobstore.BlobStore;
import org.jclouds.blobstore.BlobStoreContext;
import org.jclouds.blobstore.ContainerNotFoundException;
import org.jclouds.blobstore.KeyNotFoundException;
import org.jclouds.blobstore.domain.Blob;
import org.jclouds.blobstore.domain.BlobBuilder;
import org.jclouds.blobstore.domain.MultipartPart;
import org.jclouds.blobstore.domain.MultipartUpload;
import org.jclouds.blobstore.domain.PageSet;
import org.jclouds.blobstore.domain.StorageMetadata;
import org.jclouds.blobstore.options.CopyOptions;
import org.jclouds.blobstore.options.ListContainerOptions;
import org.jclouds.blobstore.options.PutOptions;
import org.jclouds.blobstore.strategy.internal.MultipartUploadSlicingAlgorithm;
import org.jclouds.blobstore.util.BlobUtils;
import org.jclouds.collect.Memoized;
import org.jclouds.domain.Location;
import org.jclouds.http.HttpCommand;
import org.jclouds.http.HttpRequest;
import org.jclouds.http.HttpResponse;
import org.jclouds.http.HttpResponseException;
import org.jclouds.io.ContentMetadata;
import org.jclouds.io.Payload;
import org.jclouds.io.Payloads;
import org.jclouds.io.PayloadSlicer;
import org.jclouds.util.Closeables2;
import shaded.com.google.common.annotations.Beta;
import shaded.com.google.common.annotations.VisibleForTesting;
import shaded.com.google.common.base.Predicate;
import shaded.com.google.common.base.Supplier;
import shaded.com.google.common.base.Throwables;
import shaded.com.google.common.util.concurrent.Futures;
import shaded.com.google.common.util.concurrent.ListenableFuture;
import shaded.com.google.common.util.concurrent.ListeningExecutorService;
public abstract class BaseBlobStore implements BlobStore {
protected final BlobStoreContext context;
protected final BlobUtils blobUtils;
protected final Supplier defaultLocation;
protected final Supplier> locations;
protected final PayloadSlicer slicer;
protected BaseBlobStore(BlobStoreContext context, BlobUtils blobUtils, Supplier defaultLocation,
@Memoized Supplier> locations, PayloadSlicer slicer) {
this.context = checkNotNull(context, "context");
this.blobUtils = checkNotNull(blobUtils, "blobUtils");
this.defaultLocation = checkNotNull(defaultLocation, "defaultLocation");
this.locations = checkNotNull(locations, "locations");
this.slicer = checkNotNull(slicer, "slicer");
}
@Override
public BlobStoreContext getContext() {
return context;
}
/**
* invokes {@link BlobUtilsImpl#blobBuilder }
*/
@Override
public BlobBuilder blobBuilder(String name) {
return blobUtils.blobBuilder().name(name);
}
/**
* This implementation invokes
* {@link #list(String,org.jclouds.blobstore.options.ListContainerOptions)}
*
* @param container
* container name
*/
@Override
public PageSet extends StorageMetadata> list(String container) {
return this.list(container, org.jclouds.blobstore.options.ListContainerOptions.NONE);
}
/**
* This implementation invokes {@link BlobUtilsImpl#directoryExists}
*
* @param container
* container name
* @param directory
* virtual path
*/
@Override
public boolean directoryExists(String containerName, String directory) {
return blobUtils.directoryExists(containerName, directory);
}
/**
* This implementation invokes {@link BlobUtilsImpl#createDirectory}
*
* @param container
* container name
* @param directory
* virtual path
*/
@Override
public void createDirectory(String containerName, String directory) {
blobUtils.createDirectory(containerName, directory);
}
@Override
public void removeBlobs(String container, Iterable names) {
for (String name : names) {
removeBlob(container, name);
}
}
/**
* This implementation invokes {@link #countBlobs} with the
* {@link ListContainerOptions#recursive} option.
*
* @param container
* container name
*/
@Override
public long countBlobs(String container) {
return countBlobs(container, recursive());
}
/**
* This implementation invokes {@link BlobUtilsImpl#countBlobs}
*
* @param container
* container name
*/
@Override
public long countBlobs(String containerName, ListContainerOptions options) {
return blobUtils.countBlobs(containerName, options);
}
/**
* This implementation invokes {@link #clearContainer} with the
* {@link ListContainerOptions#recursive} option.
*
* @param container
* container name
*/
@Override
public void clearContainer(String containerName) {
clearContainer(containerName, recursive());
}
/**
* This implementation invokes {@link BlobUtilsImpl#clearContainer}
*
* @param container
* container name
*/
@Override
public void clearContainer(String containerName, ListContainerOptions options) {
blobUtils.clearContainer(containerName, options);
}
/**
* This implementation invokes {@link BlobUtilsImpl#deleteDirectory}.
*
* @param container
* container name
*/
@Override
public void deleteDirectory(String containerName, String directory) {
blobUtils.deleteDirectory(containerName, directory);
}
/**
* This implementation invokes
* {@link #getBlob(String,String,org.jclouds.blobstore.options.GetOptions)}
*
* @param container
* container name
* @param key
* blob key
*/
@Override
public Blob getBlob(String container, String key) {
return getBlob(container, key, org.jclouds.blobstore.options.GetOptions.NONE);
}
/**
* This implementation invokes {@link #deleteAndEnsurePathGone}
*
* @param container
* bucket name
*/
@Override
public void deleteContainer(String container) {
deletePathAndEnsureGone(container);
}
@Override
public boolean deleteContainerIfEmpty(String container) {
try {
return deleteAndVerifyContainerGone(container);
} catch (ContainerNotFoundException cnfe) {
return true;
}
}
protected void deletePathAndEnsureGone(String path) {
checkState(retry(new Predicate() {
public boolean apply(String in) {
try {
clearContainer(in, recursive());
return deleteAndVerifyContainerGone(in);
} catch (ContainerNotFoundException e) {
return true;
}
}
}, 30000).apply(path), "%s still exists after deleting!", path);
}
@Override
public Set extends Location> listAssignableLocations() {
return locations.get();
}
/**
* Delete a container if it is empty.
*
* @param container what to delete
* @return whether container was deleted
*/
protected abstract boolean deleteAndVerifyContainerGone(String container);
@Override
public String copyBlob(String fromContainer, String fromName, String toContainer, String toName,
CopyOptions options) {
Blob blob = getBlob(fromContainer, fromName);
if (blob == null) {
throw new KeyNotFoundException(fromContainer, fromName, "while copying");
}
String eTag = blob.getMetadata().getETag();
if (eTag != null) {
eTag = maybeQuoteETag(eTag);
if (options.ifMatch() != null && !maybeQuoteETag(options.ifMatch()).equals(eTag)) {
throw returnResponseException(412);
}
if (options.ifNoneMatch() != null && maybeQuoteETag(options.ifNoneMatch()).equals(eTag)) {
throw returnResponseException(412);
}
}
Date lastModified = blob.getMetadata().getLastModified();
if (lastModified != null) {
if (options.ifModifiedSince() != null && lastModified.compareTo(options.ifModifiedSince()) <= 0) {
throw returnResponseException(412);
}
if (options.ifUnmodifiedSince() != null && lastModified.compareTo(options.ifUnmodifiedSince()) >= 0) {
throw returnResponseException(412);
}
}
InputStream is = null;
try {
is = blob.getPayload().openStream();
BlobBuilder.PayloadBlobBuilder builder = blobBuilder(toName)
.payload(is);
Long contentLength = blob.getMetadata().getContentMetadata().getContentLength();
if (contentLength != null) {
builder.contentLength(contentLength);
}
ContentMetadata metadata;
if (options.contentMetadata() != null) {
metadata = options.contentMetadata();
} else {
metadata = blob.getMetadata().getContentMetadata();
}
builder.cacheControl(metadata.getCacheControl())
.contentDisposition(metadata.getContentDisposition())
.contentEncoding(metadata.getContentEncoding())
.contentLanguage(metadata.getContentLanguage())
.contentType(metadata.getContentType());
Map userMetadata = options.userMetadata();
if (userMetadata != null) {
builder.userMetadata(userMetadata);
} else {
builder.userMetadata(blob.getMetadata().getUserMetadata());
}
return putBlob(toContainer, builder.build());
} catch (IOException ioe) {
throw Throwables.propagate(ioe);
} finally {
Closeables2.closeQuietly(is);
}
}
@shaded.com.google.inject.Inject
@Named(PROPERTY_USER_THREADS)
@VisibleForTesting
ListeningExecutorService userExecutor;
/**
* Upload using a user-provided executor, or the jclouds userExecutor
*
* @param container
* @param blob
* @param overrides
* @return the multipart blob etag
*/
@Beta
protected String putMultipartBlob(String container, Blob blob, PutOptions overrides) {
if (overrides.getUseCustomExecutor()) {
return putMultipartBlob(container, blob, overrides, overrides.getCustomExecutor());
} else {
return putMultipartBlob(container, blob, overrides, userExecutor);
}
}
@Beta
protected String putMultipartBlob(String container, Blob blob, PutOptions overrides, ListeningExecutorService executor) {
ArrayList> parts = new ArrayList>();
MultipartUpload mpu = initiateMultipartUpload(container, blob.getMetadata(), overrides);
// Cannot slice InputStream Payload since slice and close mutate the
// underlying stream. Also issue synchronous uploads to avoid buffering
// arbitrary amounts of data in-memory.
Payload payload = blob.getPayload();
boolean repeatable = blob.getPayload().isRepeatable();
if (!repeatable) {
payload = Payloads.newInputStreamPayload(new FilterInputStream((InputStream) payload.getRawContent()) {
@Override
public long skip(long offset) throws IOException {
// intentionally not implemented
return offset;
}
@Override
public void close() throws IOException {
// intentionally not implemented
}
});
}
try {
long contentLength = blob.getMetadata().getContentMetadata().getContentLength();
// TODO: inject MultipartUploadSlicingAlgorithm to override default part size
MultipartUploadSlicingAlgorithm algorithm = new MultipartUploadSlicingAlgorithm(
getMinimumMultipartPartSize(), getMaximumMultipartPartSize(), getMaximumNumberOfParts());
long partSize = algorithm.calculateChunkSize(contentLength);
int partNumber = 1;
while (partNumber <= algorithm.getParts()) {
Payload slice = slicer.slice(payload, algorithm.getCopied(), partSize);
BlobUploader b = new BlobUploader(mpu, partNumber++, slice);
parts.add(repeatable ? executor.submit(b) : Futures.immediateFuture(b.call()));
algorithm.addCopied(partSize);
}
if (algorithm.getRemaining() != 0) {
Payload slice = slicer.slice(payload, algorithm.getCopied(), algorithm.getRemaining());
BlobUploader b = new BlobUploader(mpu, partNumber, slice);
parts.add(repeatable ? executor.submit(b) : Futures.immediateFuture(b.call()));
}
return completeMultipartUpload(mpu, Futures.getUnchecked(Futures.allAsList(parts)));
} catch (RuntimeException re) {
abortMultipartUpload(mpu);
throw re;
}
}
private final class BlobUploader implements Callable {
private final MultipartUpload mpu;
private final int partNumber;
private final Payload payload;
BlobUploader(MultipartUpload mpu, int partNumber, Payload payload) {
this.mpu = mpu;
this.partNumber = partNumber;
this.payload = payload;
}
@Override
public MultipartPart call() {
return uploadMultipartPart(mpu, partNumber, payload);
}
}
private static HttpResponseException returnResponseException(int code) {
HttpResponse response = HttpResponse.builder().statusCode(code).build();
// TODO: bogus endpoint
return new HttpResponseException(new HttpCommand(HttpRequest.builder().method("GET").endpoint("http://stub")
.build()), response);
}
private static String maybeQuoteETag(String eTag) {
if (!eTag.startsWith("\"") && !eTag.endsWith("\"")) {
eTag = "\"" + eTag + "\"";
}
return eTag;
}
@Override
public void downloadBlob(String container, String name, File destination) {
throw new UnsupportedOperationException("Operation not supported yet");
}
@Override
public void downloadBlob(String container, String name, File destination, ExecutorService executor) {
throw new UnsupportedOperationException("Operation not supported yet");
}
@Override
public InputStream streamBlob(String container, String name) {
throw new UnsupportedOperationException("Operation not supported yet");
}
@Override
public InputStream streamBlob(String container, String name, ExecutorService executor) {
throw new UnsupportedOperationException("Operation not supported yet");
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy