org.apache.cassandra.repair.RepairTracker Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of cassandra-all Show documentation
Show all versions of cassandra-all Show documentation
Palantir open source project
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.apache.cassandra.repair;
import java.util.HashMap;
import java.util.Map;
import java.util.Optional;
import java.util.Set;
import java.util.stream.Collectors;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.BiMap;
import com.google.common.collect.HashBiMap;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.apache.cassandra.service.StorageServiceMBean;
import org.apache.cassandra.utils.progress.ProgressEvent;
import org.apache.cassandra.utils.progress.ProgressListener;
public class RepairTracker implements ProgressListener
{
private static final Logger logger = LoggerFactory.getLogger(RepairTracker.class);
private final BiMap argsToMostRecentRepair;
private final Map commandToProgressState;
public RepairTracker()
{
argsToMostRecentRepair = HashBiMap.create();
commandToProgressState = new HashMap<>();
}
public synchronized void track(int command, RepairArguments arguments)
{
argsToMostRecentRepair.put(arguments, command);
commandToProgressState.put(command, StorageServiceMBean.ProgressState.UNKNOWN);
}
public synchronized StorageServiceMBean.ProgressState getRepairState(int command)
{
return Optional.ofNullable(commandToProgressState.get(command))
.orElse(StorageServiceMBean.ProgressState.UNKNOWN);
}
public synchronized Optional getInProgressRepair(RepairArguments arguments)
{
return Optional.ofNullable(argsToMostRecentRepair.get(arguments))
.filter(command -> isInProgressState(commandToProgressState.get(command)));
}
public synchronized void progress(String tag, ProgressEvent event)
{
try
{
Optional repairCommand = RepairRunnable.parseCommandFromTag(tag);
if (!repairCommand.isPresent())
return;
maybeUpdateProgressState(event, repairCommand.get());
} catch (Exception e)
{
logger.error("Failed to track repair progress locally", e);
}
}
private void updateProgressState(StorageServiceMBean.ProgressState state, int command)
{
logger.info("Updating state for repair command {} to {}", command, state);
commandToProgressState.put(command, state);
if (isCompleteState(state))
{
logger.info("No longer collapsing identical repairs on top of repair command {}", command);
argsToMostRecentRepair.inverse().remove(command);
}
}
@VisibleForTesting
void maybeUpdateProgressState(ProgressEvent event, int command)
{
StorageServiceMBean.ProgressState currentState = getRepairState(command);
StorageServiceMBean.ProgressState newState;
switch (event.getType())
{
case START:
case PROGRESS:
newState = (currentState == StorageServiceMBean.ProgressState.UNKNOWN) ? StorageServiceMBean.ProgressState.IN_PROGRESS : currentState;
break;
case ABORT:
case ERROR:
newState = StorageServiceMBean.ProgressState.FAILED;
break;
case SUCCESS:
newState = (currentState == StorageServiceMBean.ProgressState.FAILED) ? currentState : StorageServiceMBean.ProgressState.SUCCEEDED;
break;
case COMPLETE:
// Something is wrong if we get a COMPLETE notification when we haven't failed or succeeded
newState = isCompleteState(currentState) ? currentState : StorageServiceMBean.ProgressState.UNKNOWN;
break;
case NOTIFICATION:
return;
default:
logger.error("Unrecognized ProgressEventType. Setting ProgressState to UNKNOWN for repair command {}",
command);
newState = StorageServiceMBean.ProgressState.UNKNOWN;
}
if (newState != currentState)
updateProgressState(newState, command);
}
private boolean isCompleteState(StorageServiceMBean.ProgressState state)
{
return state == StorageServiceMBean.ProgressState.FAILED || state == StorageServiceMBean.ProgressState.SUCCEEDED;
}
@VisibleForTesting
boolean isInProgressState(StorageServiceMBean.ProgressState state)
{
return state == StorageServiceMBean.ProgressState.IN_PROGRESS;
}
@VisibleForTesting
Map getArgsToMostRecentRepair()
{
return argsToMostRecentRepair;
}
@VisibleForTesting
Map getCommandToProgressState()
{
return commandToProgressState;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy