HDFS-12518. Re-encryption should handle task cancellation and progress better.
This commit is contained in:
parent
14b3c2695b
commit
248d9b6fff
@ -380,10 +380,16 @@ private static ZoneEncryptionInfoProto getZoneEncryptionInfoProto(
|
|||||||
static void saveFileXAttrsForBatch(FSDirectory fsd,
|
static void saveFileXAttrsForBatch(FSDirectory fsd,
|
||||||
List<FileEdekInfo> batch) {
|
List<FileEdekInfo> batch) {
|
||||||
assert fsd.getFSNamesystem().hasWriteLock();
|
assert fsd.getFSNamesystem().hasWriteLock();
|
||||||
|
assert !fsd.hasWriteLock();
|
||||||
if (batch != null && !batch.isEmpty()) {
|
if (batch != null && !batch.isEmpty()) {
|
||||||
for (FileEdekInfo entry : batch) {
|
for (FileEdekInfo entry : batch) {
|
||||||
final INode inode = fsd.getInode(entry.getInodeId());
|
final INode inode = fsd.getInode(entry.getInodeId());
|
||||||
Preconditions.checkNotNull(inode);
|
// no dir lock, so inode could be removed. no-op if so.
|
||||||
|
if (inode == null) {
|
||||||
|
NameNode.LOG.info("Cannot find inode {}, skip saving xattr for"
|
||||||
|
+ " re-encryption", entry.getInodeId());
|
||||||
|
continue;
|
||||||
|
}
|
||||||
fsd.getEditLog().logSetXAttrs(inode.getFullPathName(),
|
fsd.getEditLog().logSetXAttrs(inode.getFullPathName(),
|
||||||
inode.getXAttrFeature().getXAttrs(), false);
|
inode.getXAttrFeature().getXAttrs(), false);
|
||||||
}
|
}
|
||||||
|
@ -45,11 +45,11 @@
|
|||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.security.GeneralSecurityException;
|
import java.security.GeneralSecurityException;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
import java.util.HashMap;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Map;
|
import java.util.Map;
|
||||||
import java.util.concurrent.BlockingQueue;
|
import java.util.concurrent.BlockingQueue;
|
||||||
import java.util.concurrent.Callable;
|
import java.util.concurrent.Callable;
|
||||||
import java.util.concurrent.ConcurrentHashMap;
|
|
||||||
import java.util.concurrent.ExecutorCompletionService;
|
import java.util.concurrent.ExecutorCompletionService;
|
||||||
import java.util.concurrent.ExecutorService;
|
import java.util.concurrent.ExecutorService;
|
||||||
import java.util.concurrent.Executors;
|
import java.util.concurrent.Executors;
|
||||||
@ -112,8 +112,7 @@ public class ReencryptionHandler implements Runnable {
|
|||||||
private ExecutorCompletionService<ReencryptionTask> batchService;
|
private ExecutorCompletionService<ReencryptionTask> batchService;
|
||||||
private BlockingQueue<Runnable> taskQueue;
|
private BlockingQueue<Runnable> taskQueue;
|
||||||
// protected by ReencryptionHandler object lock
|
// protected by ReencryptionHandler object lock
|
||||||
private final Map<Long, ZoneSubmissionTracker> submissions =
|
private final Map<Long, ZoneSubmissionTracker> submissions = new HashMap<>();
|
||||||
new ConcurrentHashMap<>();
|
|
||||||
|
|
||||||
// The current batch that the handler is working on. Handler is designed to
|
// The current batch that the handler is working on. Handler is designed to
|
||||||
// be single-threaded, see class javadoc for more details.
|
// be single-threaded, see class javadoc for more details.
|
||||||
@ -132,9 +131,11 @@ public class ReencryptionHandler implements Runnable {
|
|||||||
*/
|
*/
|
||||||
void stopThreads() {
|
void stopThreads() {
|
||||||
assert dir.hasWriteLock();
|
assert dir.hasWriteLock();
|
||||||
|
synchronized (this) {
|
||||||
for (ZoneSubmissionTracker zst : submissions.values()) {
|
for (ZoneSubmissionTracker zst : submissions.values()) {
|
||||||
zst.cancelAllTasks();
|
zst.cancelAllTasks();
|
||||||
}
|
}
|
||||||
|
}
|
||||||
if (updaterExecutor != null) {
|
if (updaterExecutor != null) {
|
||||||
updaterExecutor.shutdownNow();
|
updaterExecutor.shutdownNow();
|
||||||
}
|
}
|
||||||
@ -269,33 +270,34 @@ void cancelZone(final long zoneId, final String zoneName) throws IOException {
|
|||||||
throw new IOException("Zone " + zoneName + " is not under re-encryption");
|
throw new IOException("Zone " + zoneName + " is not under re-encryption");
|
||||||
}
|
}
|
||||||
zs.cancel();
|
zs.cancel();
|
||||||
ZoneSubmissionTracker zst = submissions.get(zoneId);
|
removeZoneTrackerStopTasks(zoneId);
|
||||||
if (zst != null) {
|
|
||||||
zst.cancelAllTasks();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void removeZone(final long zoneId) {
|
void removeZone(final long zoneId) {
|
||||||
assert dir.hasWriteLock();
|
assert dir.hasWriteLock();
|
||||||
LOG.info("Removing zone {} from re-encryption.", zoneId);
|
LOG.info("Removing zone {} from re-encryption.", zoneId);
|
||||||
ZoneSubmissionTracker zst = submissions.get(zoneId);
|
removeZoneTrackerStopTasks(zoneId);
|
||||||
if (zst != null) {
|
|
||||||
zst.cancelAllTasks();
|
|
||||||
}
|
|
||||||
submissions.remove(zoneId);
|
|
||||||
getReencryptionStatus().removeZone(zoneId);
|
getReencryptionStatus().removeZone(zoneId);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
synchronized private void removeZoneTrackerStopTasks(final long zoneId) {
|
||||||
|
final ZoneSubmissionTracker zst = submissions.get(zoneId);
|
||||||
|
if (zst != null) {
|
||||||
|
zst.cancelAllTasks();
|
||||||
|
submissions.remove(zoneId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
ZoneSubmissionTracker getTracker(final long zoneId) {
|
ZoneSubmissionTracker getTracker(final long zoneId) {
|
||||||
dir.hasReadLock();
|
assert dir.hasReadLock();
|
||||||
return unprotectedGetTracker(zoneId);
|
return unprotectedGetTracker(zoneId);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* get the tracker without holding the FSDirectory lock. This is only used for
|
* Get the tracker without holding the FSDirectory lock.
|
||||||
* testing, when updater checks about pausing.
|
* The submissions object is protected by object lock.
|
||||||
*/
|
*/
|
||||||
ZoneSubmissionTracker unprotectedGetTracker(final long zoneId) {
|
synchronized ZoneSubmissionTracker unprotectedGetTracker(final long zoneId) {
|
||||||
return submissions.get(zoneId);
|
return submissions.get(zoneId);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -308,17 +310,20 @@ ZoneSubmissionTracker unprotectedGetTracker(final long zoneId) {
|
|||||||
*
|
*
|
||||||
* @param zoneId
|
* @param zoneId
|
||||||
*/
|
*/
|
||||||
void addDummyTracker(final long zoneId) {
|
void addDummyTracker(final long zoneId, ZoneSubmissionTracker zst) {
|
||||||
assert dir.hasReadLock();
|
assert dir.hasReadLock();
|
||||||
assert !submissions.containsKey(zoneId);
|
if (zst == null) {
|
||||||
final ZoneSubmissionTracker zst = new ZoneSubmissionTracker();
|
zst = new ZoneSubmissionTracker();
|
||||||
|
}
|
||||||
zst.setSubmissionDone();
|
zst.setSubmissionDone();
|
||||||
|
|
||||||
Future future = batchService.submit(
|
final Future future = batchService.submit(
|
||||||
new EDEKReencryptCallable(zoneId, new ReencryptionBatch(), this));
|
new EDEKReencryptCallable(zoneId, new ReencryptionBatch(), this));
|
||||||
zst.addTask(future);
|
zst.addTask(future);
|
||||||
|
synchronized (this) {
|
||||||
submissions.put(zoneId, zst);
|
submissions.put(zoneId, zst);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Main loop. It takes at most 1 zone per scan, and executes until the zone
|
* Main loop. It takes at most 1 zone per scan, and executes until the zone
|
||||||
@ -351,6 +356,8 @@ public void run() {
|
|||||||
}
|
}
|
||||||
LOG.info("Executing re-encrypt commands on zone {}. Current zones:{}",
|
LOG.info("Executing re-encrypt commands on zone {}. Current zones:{}",
|
||||||
zoneId, getReencryptionStatus());
|
zoneId, getReencryptionStatus());
|
||||||
|
getReencryptionStatus().markZoneStarted(zoneId);
|
||||||
|
resetSubmissionTracker(zoneId);
|
||||||
} finally {
|
} finally {
|
||||||
dir.readUnlock();
|
dir.readUnlock();
|
||||||
}
|
}
|
||||||
@ -392,7 +399,6 @@ void reencryptEncryptionZone(final long zoneId)
|
|||||||
|
|
||||||
readLock();
|
readLock();
|
||||||
try {
|
try {
|
||||||
getReencryptionStatus().markZoneStarted(zoneId);
|
|
||||||
zoneNode = dir.getInode(zoneId);
|
zoneNode = dir.getInode(zoneId);
|
||||||
// start re-encrypting the zone from the beginning
|
// start re-encrypting the zone from the beginning
|
||||||
if (zoneNode == null) {
|
if (zoneNode == null) {
|
||||||
@ -428,6 +434,20 @@ void reencryptEncryptionZone(final long zoneId)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reset the zone submission tracker for re-encryption.
|
||||||
|
* @param zoneId
|
||||||
|
*/
|
||||||
|
synchronized private void resetSubmissionTracker(final long zoneId) {
|
||||||
|
ZoneSubmissionTracker zst = submissions.get(zoneId);
|
||||||
|
if (zst == null) {
|
||||||
|
zst = new ZoneSubmissionTracker();
|
||||||
|
submissions.put(zoneId, zst);
|
||||||
|
} else {
|
||||||
|
zst.reset();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
List<XAttr> completeReencryption(final INode zoneNode) throws IOException {
|
List<XAttr> completeReencryption(final INode zoneNode) throws IOException {
|
||||||
assert dir.hasWriteLock();
|
assert dir.hasWriteLock();
|
||||||
assert dir.getFSNamesystem().hasWriteLock();
|
assert dir.getFSNamesystem().hasWriteLock();
|
||||||
@ -437,8 +457,9 @@ List<XAttr> completeReencryption(final INode zoneNode) throws IOException {
|
|||||||
LOG.info("Re-encryption completed on zone {}. Re-encrypted {} files,"
|
LOG.info("Re-encryption completed on zone {}. Re-encrypted {} files,"
|
||||||
+ " failures encountered: {}.", zoneNode.getFullPathName(),
|
+ " failures encountered: {}.", zoneNode.getFullPathName(),
|
||||||
zs.getFilesReencrypted(), zs.getNumReencryptionFailures());
|
zs.getFilesReencrypted(), zs.getNumReencryptionFailures());
|
||||||
// This also removes the zone from reencryptionStatus
|
synchronized (this) {
|
||||||
submissions.remove(zoneId);
|
submissions.remove(zoneId);
|
||||||
|
}
|
||||||
return FSDirEncryptionZoneOp
|
return FSDirEncryptionZoneOp
|
||||||
.updateReencryptionFinish(dir, INodesInPath.fromINode(zoneNode), zs);
|
.updateReencryptionFinish(dir, INodesInPath.fromINode(zoneNode), zs);
|
||||||
}
|
}
|
||||||
@ -562,11 +583,14 @@ private void submitCurrentBatch(final long zoneId)
|
|||||||
if (currentBatch.isEmpty()) {
|
if (currentBatch.isEmpty()) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ZoneSubmissionTracker zst = submissions.get(zoneId);
|
ZoneSubmissionTracker zst;
|
||||||
|
synchronized (this) {
|
||||||
|
zst = submissions.get(zoneId);
|
||||||
if (zst == null) {
|
if (zst == null) {
|
||||||
zst = new ZoneSubmissionTracker();
|
zst = new ZoneSubmissionTracker();
|
||||||
submissions.put(zoneId, zst);
|
submissions.put(zoneId, zst);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
Future future = batchService
|
Future future = batchService
|
||||||
.submit(new EDEKReencryptCallable(zoneId, currentBatch, this));
|
.submit(new EDEKReencryptCallable(zoneId, currentBatch, this));
|
||||||
zst.addTask(future);
|
zst.addTask(future);
|
||||||
@ -821,19 +845,13 @@ void throttle() throws InterruptedException {
|
|||||||
// 2. if tasks are piling up on the updater, don't create new callables
|
// 2. if tasks are piling up on the updater, don't create new callables
|
||||||
// until the queue size goes down.
|
// until the queue size goes down.
|
||||||
final int maxTasksPiled = Runtime.getRuntime().availableProcessors() * 2;
|
final int maxTasksPiled = Runtime.getRuntime().availableProcessors() * 2;
|
||||||
int totalTasks = 0;
|
int numTasks = numTasksSubmitted();
|
||||||
for (ZoneSubmissionTracker zst : submissions.values()) {
|
if (numTasks >= maxTasksPiled) {
|
||||||
totalTasks += zst.getTasks().size();
|
|
||||||
}
|
|
||||||
if (totalTasks >= maxTasksPiled) {
|
|
||||||
LOG.debug("Re-encryption handler throttling because total tasks pending"
|
LOG.debug("Re-encryption handler throttling because total tasks pending"
|
||||||
+ " re-encryption updater is {}", totalTasks);
|
+ " re-encryption updater is {}", numTasks);
|
||||||
while (totalTasks >= maxTasksPiled) {
|
while (numTasks >= maxTasksPiled) {
|
||||||
Thread.sleep(500);
|
Thread.sleep(500);
|
||||||
totalTasks = 0;
|
numTasks = numTasksSubmitted();
|
||||||
for (ZoneSubmissionTracker zst : submissions.values()) {
|
|
||||||
totalTasks += zst.getTasks().size();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -864,6 +882,14 @@ void throttle() throws InterruptedException {
|
|||||||
throttleTimerLocked.reset();
|
throttleTimerLocked.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private synchronized int numTasksSubmitted() {
|
||||||
|
int ret = 0;
|
||||||
|
for (ZoneSubmissionTracker zst : submissions.values()) {
|
||||||
|
ret += zst.getTasks().size();
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Process an Inode for re-encryption. Add to current batch if it's a file,
|
* Process an Inode for re-encryption. Add to current batch if it's a file,
|
||||||
* no-op otherwise.
|
* no-op otherwise.
|
||||||
@ -877,7 +903,7 @@ void throttle() throws InterruptedException {
|
|||||||
*/
|
*/
|
||||||
private boolean reencryptINode(final INode inode, final String ezKeyVerName)
|
private boolean reencryptINode(final INode inode, final String ezKeyVerName)
|
||||||
throws IOException, InterruptedException {
|
throws IOException, InterruptedException {
|
||||||
dir.hasReadLock();
|
assert dir.hasReadLock();
|
||||||
if (LOG.isTraceEnabled()) {
|
if (LOG.isTraceEnabled()) {
|
||||||
LOG.trace("Processing {} for re-encryption", inode.getFullPathName());
|
LOG.trace("Processing {} for re-encryption", inode.getFullPathName());
|
||||||
}
|
}
|
||||||
|
@ -93,6 +93,13 @@ static final class ZoneSubmissionTracker {
|
|||||||
numFutureDone = 0;
|
numFutureDone = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void reset() {
|
||||||
|
submissionDone = false;
|
||||||
|
tasks.clear();
|
||||||
|
numCheckpointed = 0;
|
||||||
|
numFutureDone = 0;
|
||||||
|
}
|
||||||
|
|
||||||
LinkedList<Future> getTasks() {
|
LinkedList<Future> getTasks() {
|
||||||
return tasks;
|
return tasks;
|
||||||
}
|
}
|
||||||
@ -238,12 +245,12 @@ boolean isRunning() {
|
|||||||
void markZoneSubmissionDone(final long zoneId)
|
void markZoneSubmissionDone(final long zoneId)
|
||||||
throws IOException, InterruptedException {
|
throws IOException, InterruptedException {
|
||||||
final ZoneSubmissionTracker tracker = handler.getTracker(zoneId);
|
final ZoneSubmissionTracker tracker = handler.getTracker(zoneId);
|
||||||
if (tracker != null) {
|
if (tracker != null && !tracker.getTasks().isEmpty()) {
|
||||||
tracker.submissionDone = true;
|
tracker.submissionDone = true;
|
||||||
} else {
|
} else {
|
||||||
// Caller thinks submission is done, but no tasks submitted - meaning
|
// Caller thinks submission is done, but no tasks submitted - meaning
|
||||||
// no files in the EZ need to be re-encrypted. Complete directly.
|
// no files in the EZ need to be re-encrypted. Complete directly.
|
||||||
handler.addDummyTracker(zoneId);
|
handler.addDummyTracker(zoneId, tracker);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -289,6 +296,7 @@ private void processTaskEntries(final String zoneNodePath,
|
|||||||
LOG.debug(
|
LOG.debug(
|
||||||
"Updating file xattrs for re-encrypting zone {}," + " starting at {}",
|
"Updating file xattrs for re-encrypting zone {}," + " starting at {}",
|
||||||
zoneNodePath, task.batch.getFirstFilePath());
|
zoneNodePath, task.batch.getFirstFilePath());
|
||||||
|
final int batchSize = task.batch.size();
|
||||||
for (Iterator<FileEdekInfo> it = task.batch.getBatch().iterator();
|
for (Iterator<FileEdekInfo> it = task.batch.getBatch().iterator();
|
||||||
it.hasNext();) {
|
it.hasNext();) {
|
||||||
FileEdekInfo entry = it.next();
|
FileEdekInfo entry = it.next();
|
||||||
@ -342,7 +350,7 @@ private void processTaskEntries(final String zoneNodePath,
|
|||||||
}
|
}
|
||||||
|
|
||||||
LOG.info("Updated xattrs on {}({}) files in zone {} for re-encryption,"
|
LOG.info("Updated xattrs on {}({}) files in zone {} for re-encryption,"
|
||||||
+ " starting:{}.", task.numFilesUpdated, task.batch.size(),
|
+ " starting:{}.", task.numFilesUpdated, batchSize,
|
||||||
zoneNodePath, task.batch.getFirstFilePath());
|
zoneNodePath, task.batch.getFirstFilePath());
|
||||||
}
|
}
|
||||||
task.processed = true;
|
task.processed = true;
|
||||||
@ -377,6 +385,9 @@ private List<XAttr> processCheckpoints(final INode zoneNode,
|
|||||||
ListIterator<Future> iter = tasks.listIterator();
|
ListIterator<Future> iter = tasks.listIterator();
|
||||||
while (iter.hasNext()) {
|
while (iter.hasNext()) {
|
||||||
Future<ReencryptionTask> curr = iter.next();
|
Future<ReencryptionTask> curr = iter.next();
|
||||||
|
if (curr.isCancelled()) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
if (!curr.isDone() || !curr.get().processed) {
|
if (!curr.isDone() || !curr.get().processed) {
|
||||||
// still has earlier tasks not completed, skip here.
|
// still has earlier tasks not completed, skip here.
|
||||||
break;
|
break;
|
||||||
@ -411,12 +422,12 @@ private void takeAndProcessTasks() throws Exception {
|
|||||||
final Future<ReencryptionTask> completed = batchService.take();
|
final Future<ReencryptionTask> completed = batchService.take();
|
||||||
throttle();
|
throttle();
|
||||||
checkPauseForTesting();
|
checkPauseForTesting();
|
||||||
ReencryptionTask task = completed.get();
|
|
||||||
if (completed.isCancelled()) {
|
if (completed.isCancelled()) {
|
||||||
LOG.debug("Skipped canceled re-encryption task for zone {}, last: {}",
|
// Ignore canceled zones. The cancellation is edit-logged by the handler.
|
||||||
task.zoneId, task.lastFile);
|
LOG.debug("Skipped a canceled re-encryption task");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
final ReencryptionTask task = completed.get();
|
||||||
|
|
||||||
boolean shouldRetry;
|
boolean shouldRetry;
|
||||||
do {
|
do {
|
||||||
@ -465,7 +476,11 @@ private void processTask(ReencryptionTask task)
|
|||||||
task.batch.size(), task.batch.getFirstFilePath());
|
task.batch.size(), task.batch.getFirstFilePath());
|
||||||
final ZoneSubmissionTracker tracker =
|
final ZoneSubmissionTracker tracker =
|
||||||
handler.getTracker(zoneNode.getId());
|
handler.getTracker(zoneNode.getId());
|
||||||
Preconditions.checkNotNull(tracker, "zone tracker not found " + zonePath);
|
if (tracker == null) {
|
||||||
|
// re-encryption canceled.
|
||||||
|
LOG.info("Re-encryption was canceled.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
tracker.numFutureDone++;
|
tracker.numFutureDone++;
|
||||||
EncryptionFaultInjector.getInstance().reencryptUpdaterProcessOneTask();
|
EncryptionFaultInjector.getInstance().reencryptUpdaterProcessOneTask();
|
||||||
processTaskEntries(zonePath, task);
|
processTaskEntries(zonePath, task);
|
||||||
|
@ -1467,8 +1467,8 @@ public void testReencryptCancel() throws Exception {
|
|||||||
assertEquals(5, getZoneStatus(zone.toString()).getFilesReencrypted());
|
assertEquals(5, getZoneStatus(zone.toString()).getFilesReencrypted());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Test
|
private void cancelFutureDuringReencryption(final Path zone)
|
||||||
public void testCancelFuture() throws Exception {
|
throws Exception {
|
||||||
final AtomicBoolean callableRunning = new AtomicBoolean(false);
|
final AtomicBoolean callableRunning = new AtomicBoolean(false);
|
||||||
class MyInjector extends EncryptionFaultInjector {
|
class MyInjector extends EncryptionFaultInjector {
|
||||||
private volatile int exceptionCount = 0;
|
private volatile int exceptionCount = 0;
|
||||||
@ -1498,8 +1498,6 @@ public void reencryptEncryptedKeys() throws IOException {
|
|||||||
* /dir/f
|
* /dir/f
|
||||||
*/
|
*/
|
||||||
final int len = 8196;
|
final int len = 8196;
|
||||||
final Path zoneParent = new Path("/zones");
|
|
||||||
final Path zone = new Path(zoneParent, "zone");
|
|
||||||
fsWrapper.mkdir(zone, FsPermission.getDirDefault(), true);
|
fsWrapper.mkdir(zone, FsPermission.getDirDefault(), true);
|
||||||
dfsAdmin.createEncryptionZone(zone, TEST_KEY, NO_TRASH);
|
dfsAdmin.createEncryptionZone(zone, TEST_KEY, NO_TRASH);
|
||||||
for (int i = 0; i < 10; ++i) {
|
for (int i = 0; i < 10; ++i) {
|
||||||
@ -1548,6 +1546,59 @@ public Boolean get() {
|
|||||||
assertTrue(getUpdater().isRunning());
|
assertTrue(getUpdater().isRunning());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCancelFutureThenReencrypt() throws Exception {
|
||||||
|
final Path zoneParent = new Path("/zones");
|
||||||
|
final Path zone = new Path(zoneParent, "zone");
|
||||||
|
cancelFutureDuringReencryption(zone);
|
||||||
|
|
||||||
|
// make sure new re-encryption after cancellation works.
|
||||||
|
getEzManager().resumeReencryptForTesting();
|
||||||
|
dfsAdmin.reencryptEncryptionZone(zone, ReencryptAction.START);
|
||||||
|
waitForZoneCompletes(zone.toString());
|
||||||
|
final RemoteIterator<ZoneReencryptionStatus> it =
|
||||||
|
dfsAdmin.listReencryptionStatus();
|
||||||
|
final ZoneReencryptionStatus zs = it.next();
|
||||||
|
assertEquals(zone.toString(), zs.getZoneName());
|
||||||
|
assertEquals(ZoneReencryptionStatus.State.Completed, zs.getState());
|
||||||
|
assertFalse(zs.isCanceled());
|
||||||
|
assertTrue(zs.getCompletionTime() > 0);
|
||||||
|
assertTrue(zs.getCompletionTime() > zs.getSubmissionTime());
|
||||||
|
assertEquals(10, zs.getFilesReencrypted());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void testCancelFutureThenRestart() throws Exception {
|
||||||
|
final Path zoneParent = new Path("/zones");
|
||||||
|
final Path zone = new Path(zoneParent, "zone");
|
||||||
|
cancelFutureDuringReencryption(zone);
|
||||||
|
|
||||||
|
// restart, and check status.
|
||||||
|
restartClusterDisableReencrypt();
|
||||||
|
RemoteIterator<ZoneReencryptionStatus> it =
|
||||||
|
dfsAdmin.listReencryptionStatus();
|
||||||
|
ZoneReencryptionStatus zs = it.next();
|
||||||
|
assertEquals(zone.toString(), zs.getZoneName());
|
||||||
|
assertEquals(ZoneReencryptionStatus.State.Completed, zs.getState());
|
||||||
|
assertTrue(zs.isCanceled());
|
||||||
|
assertTrue(zs.getCompletionTime() > 0);
|
||||||
|
assertTrue(zs.getCompletionTime() > zs.getSubmissionTime());
|
||||||
|
assertEquals(0, zs.getFilesReencrypted());
|
||||||
|
|
||||||
|
// verify re-encryption works after restart.
|
||||||
|
getEzManager().resumeReencryptForTesting();
|
||||||
|
dfsAdmin.reencryptEncryptionZone(zone, ReencryptAction.START);
|
||||||
|
waitForZoneCompletes(zone.toString());
|
||||||
|
it = dfsAdmin.listReencryptionStatus();
|
||||||
|
zs = it.next();
|
||||||
|
assertEquals(zone.toString(), zs.getZoneName());
|
||||||
|
assertEquals(ZoneReencryptionStatus.State.Completed, zs.getState());
|
||||||
|
assertFalse(zs.isCanceled());
|
||||||
|
assertTrue(zs.getCompletionTime() > 0);
|
||||||
|
assertTrue(zs.getCompletionTime() > zs.getSubmissionTime());
|
||||||
|
assertEquals(10, zs.getFilesReencrypted());
|
||||||
|
}
|
||||||
|
|
||||||
@Test
|
@Test
|
||||||
public void testReencryptCancelForUpdater() throws Exception {
|
public void testReencryptCancelForUpdater() throws Exception {
|
||||||
/* Setup test dir:
|
/* Setup test dir:
|
||||||
|
Loading…
Reference in New Issue
Block a user