com.emc.mongoose.base.load.step.client.LoadStepClientBase 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.load.step.client;
import static com.emc.mongoose.base.Constants.KEY_CLASS_NAME;
import static com.emc.mongoose.base.Constants.KEY_STEP_ID;
import static com.emc.mongoose.base.Exceptions.throwUncheckedIfInterrupted;
import static com.emc.mongoose.base.config.ConfigUtil.flatten;
import static com.github.akurilov.commons.lang.Exceptions.throwUnchecked;
import static org.apache.logging.log4j.CloseableThreadContext.put;
import com.emc.mongoose.base.config.AliasingUtil;
import com.emc.mongoose.base.data.DataInput;
import com.emc.mongoose.base.env.Extension;
import com.emc.mongoose.base.config.IllegalConfigurationException;
import com.emc.mongoose.base.item.io.ItemInputFactory;
import com.emc.mongoose.base.item.op.OpType;
import com.emc.mongoose.base.load.step.LoadStep;
import com.emc.mongoose.base.load.step.LoadStepBase;
import com.emc.mongoose.base.load.step.LoadStepFactory;
import com.emc.mongoose.base.load.step.client.metrics.MetricsAggregator;
import com.emc.mongoose.base.load.step.client.metrics.MetricsAggregatorImpl;
import com.emc.mongoose.base.load.step.file.FileManager;
import com.emc.mongoose.base.logging.LogUtil;
import com.emc.mongoose.base.logging.Loggers;
import com.emc.mongoose.base.metrics.MetricsManager;
import com.emc.mongoose.base.metrics.context.DistributedMetricsContext;
import com.emc.mongoose.base.metrics.context.DistributedMetricsContextImpl;
import com.emc.mongoose.base.metrics.snapshot.AllMetricsSnapshot;
import com.emc.mongoose.base.storage.driver.StorageDriver;
import com.github.akurilov.commons.net.NetUtil;
import com.github.akurilov.commons.reflection.TypeUtil;
import com.github.akurilov.commons.system.SizeInBytes;
import com.github.akurilov.confuse.Config;
import com.github.akurilov.confuse.impl.BasicConfig;
import java.io.IOException;
import java.rmi.RemoteException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;
import org.apache.logging.log4j.Level;
public abstract class LoadStepClientBase extends LoadStepBase implements LoadStepClient {
private final List stepSlices = new ArrayList<>();
private final List fileMgrs = new ArrayList<>();
// for the core configuration options which are using the files
private final List itemDataInputFileSlicers = new ArrayList<>();
private final List itemInputFileSlicers = new ArrayList<>();
private final List itemOutputFileAggregators = new ArrayList<>();
private final List opTraceLogFileAggregators = new ArrayList<>();
private final List storageAuthFileSlicers = new ArrayList<>();
public LoadStepClientBase(
final Config config,
final List extensions,
final List ctxConfigs,
final MetricsManager metricsMgr) {
super(config, extensions, ctxConfigs, metricsMgr);
}
private MetricsAggregator metricsAggregator = null;
@Override
protected final void doStartWrapped() throws IllegalArgumentException {
try (final var logCtx = put(KEY_STEP_ID, id()).put(KEY_CLASS_NAME, getClass().getSimpleName())) {
// need to set the once generated step id
config.val("load-step-id", id());
config.val("load-step-idAutoGenerated", false);
final var nodeAddrs = remoteNodeAddrs(config);
initFileManagers(nodeAddrs, fileMgrs);
final var sliceCount = 1 + nodeAddrs.size();
// init the base/shared config slices
final var configSlices = sliceConfig(config, sliceCount);
addFileClients(config, configSlices);
// init the config slices for each of the load step context configs
final var ctxConfigsSlices = (List>) new ArrayList>(sliceCount);
for (var i = 0; i < sliceCount; i++) {
ctxConfigsSlices.add(new ArrayList<>());
}
if (null != ctxConfigs) {
for (final var ctxConfig : ctxConfigs) {
final var ctxConfigSlices = sliceConfig(ctxConfig, sliceCount);
addFileClients(ctxConfig, ctxConfigSlices);
for (var i = 0; i < sliceCount; i++) {
ctxConfigsSlices.get(i).add(ctxConfigSlices.get(i));
}
}
}
initAndStartStepSlices(nodeAddrs, configSlices, ctxConfigsSlices, metricsMgr);
initAndStartMetricsAggregator();
Loggers.MSG.info(
"{}: load step client started, additional nodes: {}",
id(),
Arrays.toString(nodeAddrs.toArray()));
}
}
// determine the additional/remote full node addresses
private static List remoteNodeAddrs(final Config config) {
final var nodeConfig = config.configVal("load-step-node");
final var nodePort = nodeConfig.intVal("port");
final var nodeAddrs = nodeConfig. listVal("addrs");
return nodeAddrs == null || nodeAddrs.isEmpty()
? Collections.EMPTY_LIST
: nodeAddrs.stream()
.map(addr -> NetUtil.addPortIfMissing(addr, nodePort))
.collect(Collectors.toList());
}
private static void initFileManagers(
final List nodeAddrs, final List fileMgrsDst) {
// local file manager
fileMgrsDst.add(FileManager.INSTANCE);
// remote file managers
nodeAddrs.stream().map(FileManagerClient::resolve).forEachOrdered(fileMgrsDst::add);
}
private void addFileClients(final Config config, final List configSlices) {
final var loadConfig = config.configVal("load");
final var batchSize = loadConfig.intVal("batch-size");
final var storageConfig = config.configVal("storage");
final var itemConfig = config.configVal("item");
final var itemDataConfig = itemConfig.configVal("data");
final var verifyFlag = itemDataConfig.boolVal("verify");
final var itemDataInputConfig = itemDataConfig.configVal("input");
final var itemDataInputLayerConfig = itemDataInputConfig.configVal("layer");
final var itemDataInputLayerSizeRaw = itemDataInputLayerConfig.val("size");
final SizeInBytes itemDataLayerSize;
if (itemDataInputLayerSizeRaw instanceof String) {
itemDataLayerSize = new SizeInBytes((String) itemDataInputLayerSizeRaw);
} else {
itemDataLayerSize = new SizeInBytes(TypeUtil.typeConvert(itemDataInputLayerSizeRaw, int.class));
}
final var itemDataInputFile = itemDataInputConfig.stringVal("file");
final var itemDataInputSeed = itemDataInputConfig.stringVal("seed");
final var itemDataInputLayerCacheSize = itemDataInputLayerConfig.intVal("cache");
try (final var dataInput = DataInput.instance(
itemDataInputFile,
itemDataInputSeed,
itemDataLayerSize,
itemDataInputLayerCacheSize);
final var storageDriver = StorageDriver.instance(
extensions, storageConfig, dataInput, verifyFlag, batchSize, id());
final var itemInput = ItemInputFactory.createItemInput(itemConfig, batchSize, storageDriver)) {
if (null != itemDataInputFile && !itemDataInputFile.isEmpty()) {
itemDataInputFileSlicers.add(
new ItemDataInputFileSlicer(
id(), fileMgrs, configSlices, itemDataInputFile, batchSize));
Loggers.MSG.debug("{}: item data input file slicer initialized", id());
}
if (null != itemInput) {
itemInputFileSlicers.add(
new ItemInputFileSlicer(id(), fileMgrs, configSlices, itemInput, batchSize));
Loggers.MSG.debug("{}: item input file slicer initialized", id());
}
} catch (final IllegalConfigurationException e) {
LogUtil.exception(Level.ERROR, e, "{}: failed to init the storage driver", id());
} catch (final InterruptedException e) {
throwUnchecked(e);
} catch (final Exception e) {
LogUtil.exception(Level.WARN, e, "{}: failed to close the item input", id());
}
final var itemOutputFile = config.stringVal("item-output-file");
if (itemOutputFile != null && !itemOutputFile.isEmpty()) {
itemOutputFileAggregators.add(
new ItemOutputFileAggregator(id(), fileMgrs, configSlices, itemOutputFile));
Loggers.MSG.debug("{}: item output file aggregator initialized", id());
}
if (config.boolVal("output-metrics-trace-persist")) {
opTraceLogFileAggregators.add(new OpTraceLogFileAggregator(id(), fileMgrs));
Loggers.MSG.debug("{}: operation traces log file aggregator initialized", id());
}
final var storageAuthFile = storageConfig.stringVal("auth-file");
if (storageAuthFile != null && !storageAuthFile.isEmpty()) {
storageAuthFileSlicers.add(
new TempInputTextFileSlicer(
id(), storageAuthFile, fileMgrs, "storage-auth-file", configSlices, batchSize));
Loggers.MSG.debug("{}: storage auth file slicer initialized", id());
}
}
private void initAndStartMetricsAggregator() {
try (final var logCtx = put(KEY_STEP_ID, id()).put(KEY_CLASS_NAME, getClass().getSimpleName())) {
metricsAggregator = new MetricsAggregatorImpl(id(), stepSlices);
metricsAggregator.start();
} catch (final Exception e) {
LogUtil.exception(Level.ERROR, e, "{}: failed to start the metrics aggregator", id());
}
}
private void initAndStartStepSlices(
final List nodeAddrs,
final List configSlices,
final List> ctxConfigsSlices,
final MetricsManager metricsManager) {
final String stepTypeName;
try {
stepTypeName = getTypeName();
} catch (final RemoteException e) {
throw new AssertionError(e);
}
final var sliceCount = configSlices.size();
for (var i = 0; i < sliceCount; i++) {
final var configSlice = configSlices.get(i);
final LoadStep stepSlice;
if (i == 0) {
stepSlice = LoadStepFactory.createLocalLoadStep(
configSlice, extensions, ctxConfigsSlices.get(i), metricsManager, stepTypeName);
} else {
final var nodeAddrWithPort = nodeAddrs.get(i - 1);
stepSlice = LoadStepSliceUtil.resolveRemote(
configSlice, ctxConfigsSlices.get(i), stepTypeName, nodeAddrWithPort);
}
stepSlices.add(stepSlice);
if (stepSlice != null) {
try {
stepSlice.start();
} catch (final Exception e) {
if (e instanceof InterruptedException) {
throwUnchecked(e);
}
LogUtil.exception(
Level.ERROR, e, "{}: failed to start the step slice \"{}\"", id(), stepSlice);
}
}
}
}
private List sliceConfig(final Config config, final int sliceCount) {
final var configSlices = (List) new ArrayList(sliceCount);
for (var i = 0; i < sliceCount; i++) {
final var configSlice = ConfigSliceUtil.initSlice(config);
if (i == 0) {
// local step slice: disable the average metrics output
configSlice.val("output-metrics-average-period", "0s");
}
configSlices.add(configSlice);
}
if (sliceCount > 1) { // distributed mode
//
final var countLimit = config.longVal("load-op-limit-count");
if (countLimit > 0) {
ConfigSliceUtil.sliceLongValue(countLimit, configSlices, "load-op-limit-count");
configSlices.stream()
.mapToLong(configSlice -> configSlice.longVal("load-op-limit-count"))
.filter(countLimitSlice -> countLimitSlice == 0)
.findAny()
.ifPresent(
countLimitSlice -> Loggers.MSG.fatal(
"{}: the count limit ({}) is too small to be sliced among the {} nodes, the load step "
+ "won't work correctly",
id(),
countLimit,
sliceCount));
}
//
final var countFailLimit = config.longVal("load-op-limit-fail-count");
if (countFailLimit > 0) {
ConfigSliceUtil.sliceLongValue(countFailLimit, configSlices, "load-op-limit-fail-count");
configSlices.stream()
.mapToLong(configSlice -> configSlice.longVal("load-op-limit-fail-count"))
.filter(failCountLimitSlice -> failCountLimitSlice == 0)
.findAny()
.ifPresent(
failCountLimitSlice -> Loggers.MSG.error(
"{}: the failures count limit ({}) is too small to be sliced among the {} nodes, the load "
+ "step may not work correctly",
id(),
countLimit,
sliceCount));
}
//
final var rateLimit = config.doubleVal("load-op-limit-rate");
if (rateLimit > 0) {
ConfigSliceUtil.sliceDoubleValue(rateLimit, configSlices, "load-op-limit-rate");
}
//
final long sizeLimit;
final var sizeLimitRaw = config.val("load-step-limit-size");
if (sizeLimitRaw instanceof String) {
sizeLimit = SizeInBytes.toFixedSize((String) sizeLimitRaw);
} else {
sizeLimit = TypeUtil.typeConvert(sizeLimitRaw, long.class);
}
if (sizeLimit > 0) {
ConfigSliceUtil.sliceLongValue(sizeLimit, configSlices, "load-step-limit-size");
}
//
try {
final var storageNetNodeConfig = config.configVal("storage-net-node");
final var sliceStorageNodesFlag = storageNetNodeConfig.boolVal("slice");
if (sliceStorageNodesFlag) {
final var storageNodeAddrs = storageNetNodeConfig. listVal("addrs");
ConfigSliceUtil.sliceStorageNodeAddrs(configSlices, storageNodeAddrs);
}
} catch (final NoSuchElementException ignore) {}
//
ConfigSliceUtil.sliceItemNaming(configSlices);
}
return configSlices;
}
private int sliceCount() {
return stepSlices.size();
}
protected final void initMetrics(
final int originIndex,
final OpType opType,
final int concurrencyLimit,
final Config metricsConfig,
final SizeInBytes itemDataSize,
final boolean outputColorFlag) {
final var concurrencyThreshold = (int) (concurrencyLimit * metricsConfig.doubleVal("threshold"));
final var metricsAvgPersistFlag = metricsConfig.boolVal("average-persist");
final var metricsSumPersistFlag = metricsConfig.boolVal("summary-persist");
// it's not known yet how many nodes are involved, so passing the function "this::sliceCount"
// reference for
// further usage
final var metricsCtx = (DistributedMetricsContext) DistributedMetricsContextImpl.builder()
.id(id())
.opType(opType)
.nodeCountSupplier(this::sliceCount)
.concurrencyLimit(concurrencyLimit)
.concurrencyThreshold(concurrencyThreshold)
.itemDataSize(itemDataSize)
.outputPeriodSec(avgPeriod(metricsConfig))
.stdOutColorFlag(outputColorFlag)
.avgPersistFlag(metricsAvgPersistFlag)
.sumPersistFlag(metricsSumPersistFlag)
.snapshotsSupplier(() -> metricsSnapshotsByIndex(originIndex))
.quantileValues(quantiles(metricsConfig))
.nodeAddrs(remoteNodeAddrs(config))
.comment(config.stringVal("run-comment"))
.build();
metricsContexts.add(metricsCtx);
}
private List quantiles(final Config metricsConfig) {
return metricsConfig.listVal("quantiles").stream()
.map(v -> Double.valueOf(v.toString()))
.collect(Collectors.toList());
}
private List metricsSnapshotsByIndex(final int originIndex) {
return metricsAggregator == null
? Collections.emptyList()
: metricsAggregator.metricsSnapshotsByIndex(originIndex);
}
@Override
protected final void doShutdown() {
stepSlices
.parallelStream()
.forEach(
stepSlice -> {
try (final var logCtx = put(KEY_STEP_ID, id()).put(KEY_CLASS_NAME, getClass().getSimpleName())) {
stepSlice.shutdown();
} catch (final RemoteException e) {
LogUtil.exception(
Level.WARN, e, "{}: failed to shutdown the step service {}", id(), stepSlice);
}
});
}
@Override
public final boolean await(final long timeout, final TimeUnit timeUnit)
throws IllegalStateException, InterruptedException {
final var stepSliceCount = stepSlices.size();
try (final var logCtx = put(KEY_STEP_ID, id()).put(KEY_CLASS_NAME, getClass().getSimpleName())) {
if (0 == stepSliceCount) {
throw new IllegalStateException("No step slices are available");
}
Loggers.MSG.debug(
"{}: await for {} step slices for at most {} {}...",
id(),
stepSliceCount,
timeout,
timeUnit.name().toLowerCase());
return stepSlices
.parallelStream()
.map(
stepSlice -> {
try {
return stepSlice.await(timeout, timeUnit);
} catch (final InterruptedException e) {
throwUnchecked(e);
} catch (final RemoteException e) {
return false;
}
return false;
})
.reduce((flag1, flag2) -> flag1 && flag2)
.orElse(false);
} finally {
Loggers.MSG.info("{}: await for {} step slices done", id(), stepSliceCount);
}
}
@Override
protected final void doStop() {
stepSlices
.parallelStream()
.forEach(
stepSlice -> {
try (final var logCtx = put(KEY_STEP_ID, stepSlice.id())
.put(KEY_CLASS_NAME, getClass().getSimpleName())) {
stepSlice.stop();
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.trace(
Loggers.ERR,
Level.WARN,
e,
"{}: failed to stop the step slice \"{}\"",
id(),
stepSlice);
}
});
if (null != metricsAggregator) {
try {
metricsAggregator.stop();
} catch (final RemoteException ignored) {}
}
super.doStop();
}
@Override
protected final void doClose() throws IOException {
try (final var logCtx = put(KEY_STEP_ID, id()).put(KEY_CLASS_NAME, getClass().getSimpleName())) {
super.doClose();
if (null != metricsAggregator) {
metricsAggregator.close();
metricsAggregator = null;
}
stepSlices
.parallelStream()
.forEach(
stepSlice -> {
try {
stepSlice.close();
Loggers.MSG.debug("{}: step slice \"{}\" closed", id(), stepSlice);
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.exception(
Level.WARN,
e,
"{}: failed to close the step service \"{}\"",
id(),
stepSlice);
}
});
Loggers.MSG.debug("{}: closed all {} step slices", id(), stepSlices.size());
stepSlices.clear();
itemDataInputFileSlicers.forEach(
itemDataInputFileSlicer -> {
try {
itemDataInputFileSlicer.close();
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.exception(
Level.WARN,
e,
"{}: failed to close the item data input file slicer \"{}\"",
id(),
itemDataInputFileSlicer);
}
});
itemDataInputFileSlicers.clear();
itemInputFileSlicers.forEach(
itemInputFileSlicer -> {
try {
itemInputFileSlicer.close();
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.exception(
Level.WARN,
e,
"{}: failed to close the item input file slicer \"{}\"",
id(),
itemInputFileSlicer);
}
});
itemInputFileSlicers.clear();
itemOutputFileAggregators
.parallelStream()
.forEach(
itemOutputFileAggregator -> {
try {
itemOutputFileAggregator.close();
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.exception(
Level.WARN,
e,
"{}: failed to close the item output file aggregator \"{}\"",
id(),
itemOutputFileAggregator);
}
});
itemOutputFileAggregators.clear();
opTraceLogFileAggregators
.parallelStream()
.forEach(
opTraceLogFileAggregator -> {
try {
opTraceLogFileAggregator.close();
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.exception(
Level.WARN,
e,
"{}: failed to close the operation traces log file aggregator \"{}\"",
id(),
opTraceLogFileAggregator);
}
});
opTraceLogFileAggregators.clear();
storageAuthFileSlicers.forEach(
storageAuthFileSlicer -> {
try {
storageAuthFileSlicer.close();
} catch (final Exception e) {
throwUncheckedIfInterrupted(e);
LogUtil.exception(
Level.WARN,
e,
"{}: failed to close the storage auth file slicer \"{}\"",
id(),
storageAuthFileSlicer);
}
});
storageAuthFileSlicers.clear();
}
}
@Override
public final T config(final Map configMap) {
if (ctxConfigs != null) {
throw new IllegalStateException("config(...) should be invoked before any append(...) call");
}
final var configCopy = (Config) new BasicConfig(config);
final var argValPairs = (Map) new HashMap();
flatten(configMap, argValPairs, config.pathSep(), null);
final var aliasingConfig = config.
© 2015 - 2025 Weber Informatics LLC | Privacy Policy