com.emc.mongoose.base.storage.driver.StorageDriverBase Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of mongoose-base Show documentation
Show all versions of mongoose-base Show documentation
Mongoose is a high-load storage performance testing tool
package com.emc.mongoose.base.storage.driver;
import static com.emc.mongoose.base.Constants.KEY_CLASS_NAME;
import static com.emc.mongoose.base.Constants.KEY_STEP_ID;
import com.emc.mongoose.base.concurrent.DaemonBase;
import com.emc.mongoose.base.data.DataInput;
import com.emc.mongoose.base.config.IllegalConfigurationException;
import com.emc.mongoose.base.item.Item;
import com.emc.mongoose.base.item.op.Operation;
import com.emc.mongoose.base.item.op.data.DataOperation;
import com.emc.mongoose.base.logging.Loggers;
import com.emc.mongoose.base.storage.Credential;
import com.github.akurilov.commons.concurrent.ThreadUtil;
import com.github.akurilov.commons.io.Input;
import com.github.akurilov.confuse.Config;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ArrayBlockingQueue;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;
import java.util.function.Function;
import org.apache.logging.log4j.CloseableThreadContext;
/** Created by kurila on 11.07.16. */
public abstract class StorageDriverBase> extends DaemonBase
implements StorageDriver {
private final DataInput itemDataInput;
protected final String stepId;
private final BlockingQueue opsResultsQueue;
protected final int concurrencyLimit;
protected final int ioWorkerCount;
protected final String namespace;
protected final Credential credential;
protected final boolean verifyFlag;
protected final ConcurrentMap pathToCredMap = new ConcurrentHashMap<>(1);
private final ConcurrentMap pathMap = new ConcurrentHashMap<>(1);
protected Function requestNewPathFunc = this::requestNewPath;
protected final ConcurrentMap authTokens = new ConcurrentHashMap<>(1);
protected Function requestAuthTokenFunc = this::requestNewAuthToken;
protected StorageDriverBase(
final String stepId,
final DataInput itemDataInput,
final Config storageConfig,
final boolean verifyFlag)
throws IllegalConfigurationException {
this.itemDataInput = itemDataInput;
final Config driverConfig = storageConfig.configVal("driver");
final Config limitConfig = driverConfig.configVal("limit");
final int outputQueueCapacity = limitConfig.intVal("queue-output");
this.opsResultsQueue = new ArrayBlockingQueue<>(outputQueueCapacity);
this.stepId = stepId;
this.namespace = storageConfig.stringVal("namespace");
final Config authConfig = storageConfig.configVal("auth");
this.credential = Credential.getInstance(authConfig.stringVal("uid"), authConfig.stringVal("secret"));
final String authToken = authConfig.stringVal("token");
if (authToken != null) {
if (this.credential == null) {
this.authTokens.put(Credential.NONE, authToken);
} else {
this.authTokens.put(credential, authToken);
}
}
this.concurrencyLimit = limitConfig.intVal("concurrency");
this.verifyFlag = verifyFlag;
final int confWorkerCount = driverConfig.intVal("threads");
if (confWorkerCount > 0) {
ioWorkerCount = confWorkerCount;
} else if (concurrencyLimit > 0) {
ioWorkerCount = Math.min(concurrencyLimit, ThreadUtil.getHardwareThreadCount());
} else {
ioWorkerCount = ThreadUtil.getHardwareThreadCount();
}
}
protected abstract String requestNewPath(final String path);
protected abstract String requestNewAuthToken(final Credential credential);
protected boolean prepare(final O op) {
op.reset();
if (op instanceof DataOperation) {
((DataOperation) op).item().dataInput(itemDataInput);
}
final String dstPath = op.dstPath();
final Credential credential = op.credential();
if (credential != null) {
pathToCredMap.putIfAbsent(dstPath == null ? "" : dstPath, credential);
if (requestAuthTokenFunc != null) {
authTokens.computeIfAbsent(credential, requestAuthTokenFunc);
}
}
if (requestNewPathFunc != null) {
// NOTE: in the distributed mode null dstPath becomes empty one
if (dstPath != null && !dstPath.isEmpty()) {
if (null == pathMap.computeIfAbsent(dstPath, requestNewPathFunc)) {
Loggers.ERR.debug("Failed to compute the destination path for the operation: {}", op);
op.status(Operation.Status.FAIL_UNKNOWN);
// return false;
}
}
}
return true;
}
protected boolean handleCompleted(final O op) {
if (isStopped()) {
return false;
} else {
if (Loggers.MSG.isTraceEnabled()) {
Loggers.MSG.trace("{}: Load operation completed", op);
}
final O opResult = op.result();
if (opsResultsQueue.offer(opResult)) {
return true;
} else {
Loggers.ERR.error(
"{}: Load operations results queue overflow, dropping the result", toString());
return false;
}
}
}
@Override
public final int concurrencyLimit() {
return concurrencyLimit;
}
@Override
public final O get() {
return opsResultsQueue.poll();
}
@Override
public final int get(final List buffer, final int limit) {
return opsResultsQueue.drainTo(buffer, limit);
}
@Override
public final long skip(final long count) {
int n = (int) Math.min(count, Integer.MAX_VALUE);
final List tmpBuff = new ArrayList<>(n);
n = opsResultsQueue.drainTo(tmpBuff, n);
tmpBuff.clear();
return n;
}
@Override
public final boolean hasRemainingResults() {
return !opsResultsQueue.isEmpty();
}
@Override
public Input getInput() {
return this;
}
@Override
protected void doClose() throws IOException, IllegalStateException {
try (final CloseableThreadContext.Instance logCtx = CloseableThreadContext.put(KEY_STEP_ID, stepId)
.put(KEY_CLASS_NAME, StorageDriverBase.class.getSimpleName())) {
itemDataInput.close();
final int opResultsQueueSize = opsResultsQueue.size();
if (opResultsQueueSize > 0) {
Loggers.ERR.warn(
"{}: Load operations results queue contains {} unhandled elements",
toString(),
opResultsQueueSize);
}
opsResultsQueue.clear();
authTokens.clear();
pathToCredMap.clear();
pathMap.clear();
super.doClose();
Loggers.MSG.debug("{}: closed", toString());
}
}
@Override
public String toString() {
return "storage/driver/" + concurrencyLimit + "/%s/" + hashCode();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy