blob: 88743a216676c963523f2556980747f8aa516872 [file] [log] [blame]
/*
* 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.strategy.internal;
import static org.easymock.EasyMock.createMock;
import static org.easymock.EasyMock.createMockBuilder;
import static org.easymock.EasyMock.createControl;
import static org.easymock.EasyMock.isA;
import static org.easymock.EasyMock.replay;
import static org.testng.Assert.assertEquals;
import static org.testng.Assert.assertTrue;
import static org.testng.Assert.fail;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
import java.util.concurrent.Callable;
import java.util.concurrent.Executor;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicBoolean;
import org.easymock.EasyMock;
import org.easymock.IMocksControl;
import org.jclouds.ContextBuilder;
import org.jclouds.blobstore.BlobStore;
import org.jclouds.blobstore.ContainerNotFoundException;
import org.jclouds.blobstore.options.ListContainerOptions;
import org.jclouds.util.Closeables2;
import org.jclouds.blobstore.domain.PageSet;
import org.jclouds.blobstore.domain.StorageMetadata;
import org.jclouds.blobstore.internal.BlobRuntimeException;
import org.jclouds.http.handlers.BackoffLimitedRetryHandler;
import org.testng.annotations.AfterMethod;
import org.testng.annotations.BeforeMethod;
import org.testng.annotations.Test;
import com.google.common.util.concurrent.Futures;
import com.google.common.util.concurrent.ListenableFuture;
import com.google.common.util.concurrent.ListeningExecutorService;
import com.google.inject.Injector;
@Test(testName = "DeleteAllKeysInListTest", singleThreaded = true)
public class DeleteAllKeysInListTest {
private BlobStore blobstore;
private DeleteAllKeysInList deleter;
private BackoffLimitedRetryHandler retryHandler;
private static final String containerName = "container";
private static final String directoryName = "directory";
private static final int maxParallelDeletes = 1024;
@BeforeMethod
void setupBlobStore() {
Injector injector = ContextBuilder.newBuilder("transient")
.buildInjector();
blobstore = injector.getInstance(BlobStore.class);
deleter = injector.getInstance(DeleteAllKeysInList.class);
retryHandler = injector.getInstance(BackoffLimitedRetryHandler.class);
createDataSet();
}
@AfterMethod
void close() {
Closeables2.closeQuietly(blobstore.getContext());
}
public void testExecuteWithoutOptionsClearsRecursively() {
deleter.execute(containerName);
assertEquals(blobstore.countBlobs(containerName), 0);
}
public void testExecuteRecursive() {
deleter.execute(containerName, ListContainerOptions.Builder.recursive());
assertEquals(blobstore.countBlobs(containerName), 0);
}
public void testExecuteNonRecursive() {
deleter.execute(containerName, ListContainerOptions.NONE);
assertEquals(blobstore.countBlobs(containerName), 2222);
}
public void testExecuteInDirectory() {
deleter.execute(containerName, ListContainerOptions.Builder.inDirectory(directoryName));
assertEquals(blobstore.countBlobs(containerName), 1111);
}
public void testContainerNotFound() {
IMocksControl mockControl = createControl();
BlobStore blobStore = mockControl.createMock(BlobStore.class);
ListeningExecutorService executorService = mockControl
.createMock(ListeningExecutorService.class);
DeleteAllKeysInList testDeleter = createMockBuilder(
DeleteAllKeysInList.class).withConstructor(executorService,
blobStore, retryHandler, maxParallelDeletes).createMock();
EasyMock.<PageSet<? extends StorageMetadata>> expect(blobStore.list(
isA(String.class), isA(ListContainerOptions.class)))
.andThrow(new ContainerNotFoundException()).once();
replay(blobStore);
testDeleter.execute(containerName,
ListContainerOptions.Builder.recursive());
// No blobs will be deleted since blobStore.list will throw a
// ContainerNotFoundException.
assertEquals(blobstore.countBlobs(containerName), 3333);
}
@SuppressWarnings("unchecked")
public void testDeleteAfterFutureFailure() {
IMocksControl mockControl = createControl();
ListeningExecutorService executorService = mockControl
.createMock(ListeningExecutorService.class);
DeleteAllKeysInList testDeleter = createMockBuilder(
DeleteAllKeysInList.class).withConstructor(executorService,
blobstore, retryHandler, maxParallelDeletes).createMock();
// Fail the first future that is created for deleting blobs.
EasyMock.<ListenableFuture<?>> expect(
executorService.submit(isA(Callable.class)))
.andReturn(
Futures.<Void> immediateFailedFuture(new RuntimeException()))
.once();
// There should be at least another 3333 calls to executorService.submit
// since there are 3333 blobs.
EasyMock.expectLastCall().andReturn(Futures.<Void> immediateFuture(null))
.times(3333, Integer.MAX_VALUE);
replay(executorService);
testDeleter.execute(containerName,
ListContainerOptions.Builder.recursive());
}
@SuppressWarnings("unchecked")
public void testExceptionThrownAfterMaxRetries() {
IMocksControl mockControl = createControl();
ListeningExecutorService executorService = mockControl
.createMock(ListeningExecutorService.class);
DeleteAllKeysInList testDeleter = createMockBuilder(
DeleteAllKeysInList.class).withConstructor(executorService,
blobstore, retryHandler, maxParallelDeletes).createMock();
// Fail the first future that is created for deleting blobs.
EasyMock.<ListenableFuture<?>> expect(
executorService.submit(isA(Callable.class)))
.andReturn(
Futures.<Void> immediateFailedFuture(new RuntimeException()))
.once();
EasyMock.expectLastCall().andReturn(Futures.<Void> immediateFuture(null))
.anyTimes();
replay(executorService);
testDeleter.setMaxErrors(1);
boolean blobRunTimeExceptionThrown = false;
try {
testDeleter.execute(containerName,
ListContainerOptions.Builder.recursive());
} catch (BlobRuntimeException be) {
blobRunTimeExceptionThrown = true;
}
assertTrue(blobRunTimeExceptionThrown, "Expected a BlobRunTimeException");
}
@SuppressWarnings("unchecked")
public void testFuturesCancelledOnFailure() {
IMocksControl mockControl = createControl();
ListeningExecutorService executorService = mockControl
.createMock(ListeningExecutorService.class);
DeleteAllKeysInList testDeleter = createMockBuilder(
DeleteAllKeysInList.class).withConstructor(executorService,
blobstore, retryHandler, maxParallelDeletes).createMock();
final AtomicBoolean deleteFailure = new AtomicBoolean();
final Semaphore semaphore = createMock(Semaphore.class);
final Set<ListenableFuture<Void>> outstandingFutures = Collections
.synchronizedSet(new HashSet<ListenableFuture<Void>>());
final ListenableFuture<Void> blobDelFuture = createMock(ListenableFuture.class);
try {
// Allow the first semaphore acquire to succeed.
EasyMock.expect(semaphore.tryAcquire(Long.MAX_VALUE,
TimeUnit.MILLISECONDS)).andReturn(true).once();
EasyMock.<ListenableFuture<?>> expect(
executorService.submit(isA(Callable.class)))
.andReturn(blobDelFuture).once();
// Fail the second semaphore acquire.
EasyMock.expect(semaphore.tryAcquire(Long.MAX_VALUE,
TimeUnit.MILLISECONDS))
.andReturn(false).anyTimes();
blobDelFuture.addListener(isA(Runnable.class), isA(Executor.class));
EasyMock.expectLastCall();
EasyMock.expect(blobDelFuture.cancel(true)).andReturn(true)
.atLeastOnce();
} catch (InterruptedException e) {
fail();
}
replay(semaphore, executorService, blobDelFuture);
testDeleter.setMaxErrors(1);
testDeleter.executeOneIteration(containerName,
ListContainerOptions.Builder.recursive(), semaphore,
outstandingFutures, deleteFailure, /* blocking = */false);
assertEquals(outstandingFutures.size(), 1);
assertTrue(deleteFailure.get());
}
/**
* Create a container "container" with 1111 blobs named "blob-%d". Create a
* subdirectory "directory" which contains 2222 more blobs named
* "directory/blob-%d".
*/
private void createDataSet() {
String blobNameFmt = "blob-%d";
String directoryBlobNameFmt = "%s/blob-%d";
blobstore.createContainerInLocation(null, containerName);
for (int i = 0; i < 1111; i++) {
String blobName = String.format(blobNameFmt, i);
blobstore.putBlob(containerName, blobstore.blobBuilder(blobName).payload(blobName).build());
}
for (int i = 0; i < 2222; i++) {
String directoryBlobName = String.format(directoryBlobNameFmt, directoryName, i);
blobstore.putBlob(containerName, blobstore.blobBuilder(directoryBlobName).payload(directoryBlobName).build());
}
assertEquals(blobstore.countBlobs(containerName), 3333);
}
}