com.emc.mongoose.load.generator.BasicLoadGeneratorBuilder Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of mongoose-load-generator Show documentation
Show all versions of mongoose-load-generator Show documentation
Mongoose is a high-load storage performance testing tool
package com.emc.mongoose.load.generator;
import com.emc.mongoose.api.common.exception.OmgShootMyFootException;
import com.emc.mongoose.api.model.concurrent.LogContextThreadFactory;
import com.github.akurilov.commons.collection.Range;
import com.github.akurilov.commons.system.SizeInBytes;
import com.emc.mongoose.api.common.supply.BatchSupplier;
import com.emc.mongoose.api.common.supply.ConstantStringSupplier;
import com.github.akurilov.commons.io.Input;
import com.emc.mongoose.api.common.supply.RangePatternDefinedSupplier;
import com.emc.mongoose.api.model.io.task.IoTask;
import com.emc.mongoose.api.model.io.task.IoTaskBuilder;
import com.emc.mongoose.api.model.io.task.data.BasicDataIoTaskBuilder;
import com.emc.mongoose.api.model.io.task.data.DataIoTaskBuilder;
import com.emc.mongoose.api.model.io.task.path.BasicPathIoTaskBuilder;
import com.emc.mongoose.api.model.io.task.token.BasicTokenIoTaskBuilder;
import com.emc.mongoose.api.model.item.BasicDataItemFactory;
import com.emc.mongoose.api.model.item.ItemNameSupplier;
import com.emc.mongoose.api.model.item.CsvFileItemInput;
import com.emc.mongoose.api.model.item.DataItem;
import com.emc.mongoose.api.model.item.TransferConvertBuffer;
import com.emc.mongoose.api.model.item.Item;
import com.emc.mongoose.api.model.item.ItemFactory;
import com.emc.mongoose.api.model.item.ItemNamingType;
import com.emc.mongoose.api.model.item.ItemType;
import com.emc.mongoose.api.model.item.NewDataItemInput;
import com.emc.mongoose.api.model.io.IoType;
import static com.emc.mongoose.api.common.Constants.M;
import static com.emc.mongoose.api.common.supply.PatternDefinedSupplier.PATTERN_CHAR;
import static com.emc.mongoose.api.model.item.DataItem.getRangeCount;
import static com.emc.mongoose.api.model.storage.StorageDriver.BUFF_SIZE_MIN;
import com.emc.mongoose.api.model.item.NewItemInput;
import com.emc.mongoose.api.model.storage.StorageDriver;
import com.emc.mongoose.ui.config.item.ItemConfig;
import com.emc.mongoose.ui.config.item.data.ranges.RangesConfig;
import com.emc.mongoose.ui.config.item.input.InputConfig;
import com.emc.mongoose.ui.config.item.naming.NamingConfig;
import com.emc.mongoose.ui.config.load.LoadConfig;
import com.emc.mongoose.ui.config.load.generator.GeneratorConfig;
import com.emc.mongoose.ui.config.load.generator.recycle.RecycleConfig;
import com.emc.mongoose.ui.config.storage.auth.AuthConfig;
import com.emc.mongoose.ui.config.test.step.limit.LimitConfig;
import com.emc.mongoose.ui.log.LogUtil;
import com.emc.mongoose.ui.log.Loggers;
import org.apache.logging.log4j.Level;
import java.io.BufferedReader;
import java.io.EOFException;
import java.io.IOException;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.rmi.RemoteException;
import java.util.ArrayList;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CancellationException;
import java.util.concurrent.Executors;
import java.util.concurrent.ScheduledExecutorService;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.LongAdder;
import java.util.stream.Collectors;
/**
Created by andrey on 12.11.16.
*/
public class BasicLoadGeneratorBuilder<
I extends Item, O extends IoTask, T extends BasicLoadGenerator
>
implements LoadGeneratorBuilder {
private ItemConfig itemConfig;
private LoadConfig loadConfig;
private LimitConfig limitConfig;
private ItemType itemType;
private ItemFactory itemFactory;
private AuthConfig authConfig;
private List> storageDrivers;
private Input itemInput = null;
private long sizeEstimate = 0;
private int batchSize;
@Override
public BasicLoadGeneratorBuilder setItemConfig(final ItemConfig itemConfig) {
this.itemConfig = itemConfig;
return this;
}
@Override
public BasicLoadGeneratorBuilder setLoadConfig(final LoadConfig loadConfig) {
this.loadConfig = loadConfig;
this.batchSize = loadConfig.getBatchConfig().getSize();
return this;
}
@Override
public BasicLoadGeneratorBuilder setLimitConfig(final LimitConfig limitConfig) {
this.limitConfig = limitConfig;
return this;
}
@Override
public BasicLoadGeneratorBuilder setItemType(final ItemType itemType) {
this.itemType = itemType;
return this;
}
@Override
public BasicLoadGeneratorBuilder setItemFactory(final ItemFactory itemFactory) {
this.itemFactory = itemFactory;
return this;
}
@Override
public BasicLoadGeneratorBuilder setAuthConfig(final AuthConfig authConfig) {
this.authConfig = authConfig;
return this;
}
@Override
public BasicLoadGeneratorBuilder setStorageDrivers(
final List> storageDrivers
) {
this.storageDrivers = storageDrivers;
return this;
}
@Override @SuppressWarnings("unchecked")
public BasicLoadGeneratorBuilder setItemInput(final Input itemInput) {
/*if(this.itemInput != null) {
try {
this.itemInput.close();
} catch(final IOException ignored) {
}
}*/
this.itemInput = itemInput;
// chain transfer buffer is not resettable
if(!(itemInput instanceof TransferConvertBuffer)) {
sizeEstimate = estimateTransferSize(
null, IoType.valueOf(loadConfig.getType().toUpperCase()), (Input) itemInput
);
}
return this;
}
@SuppressWarnings("unchecked")
public T build()
throws OmgShootMyFootException {
// prepare
final IoTaskBuilder ioTaskBuilder;
final long countLimit = limitConfig.getCount();
final SizeInBytes sizeLimit = limitConfig.getSize();
final GeneratorConfig generatorConfig = loadConfig.getGeneratorConfig();
final boolean shuffleFlag = generatorConfig.getShuffle();
final InputConfig inputConfig = itemConfig.getInputConfig();
final RangesConfig rangesConfig = itemConfig.getDataConfig().getRangesConfig();
// init the I/O task builder
if(ItemType.DATA.equals(itemType)) {
final List fixedRangesConfig = rangesConfig.getFixed();
final List fixedRanges;
if(fixedRangesConfig != null) {
fixedRanges = fixedRangesConfig
.stream()
.map(Range::new)
.collect(Collectors.toList());
} else {
fixedRanges = Collections.EMPTY_LIST;
}
ioTaskBuilder = (IoTaskBuilder) new BasicDataIoTaskBuilder()
.setFixedRanges(fixedRanges)
.setRandomRangesCount(rangesConfig.getRandom())
.setSizeThreshold(rangesConfig.getThreshold().get());
} else if(ItemType.PATH.equals(itemType)){
ioTaskBuilder = (IoTaskBuilder) new BasicPathIoTaskBuilder();
} else {
ioTaskBuilder = (IoTaskBuilder) new BasicTokenIoTaskBuilder();
}
// determine the operations type
final IoType ioType = IoType.valueOf(loadConfig.getType().toUpperCase());
ioTaskBuilder.setIoType(ioType);
// determine the input path
String itemInputPath = inputConfig.getPath();
if(itemInputPath != null && itemInputPath.indexOf('/') != 0) {
itemInputPath = '/' + itemInputPath;
}
ioTaskBuilder.setInputPath(itemInputPath);
// determine the output path
final BatchSupplier outputPathSupplier;
if(IoType.CREATE.equals(ioType) && ItemType.DATA.equals(itemType)) {
outputPathSupplier = getOutputPathSupplier();
} else {
outputPathSupplier = null;
}
ioTaskBuilder.setOutputPathSupplier(outputPathSupplier);
// init the credentials, multi-user case support
final BatchSupplier uidSupplier;
final String uid = authConfig.getUid();
if(uid == null) {
uidSupplier = null;
} else if(-1 != uid.indexOf(PATTERN_CHAR)) {
uidSupplier = new RangePatternDefinedSupplier(uid);
} else {
uidSupplier = new ConstantStringSupplier(uid);
}
ioTaskBuilder.setUidSupplier(uidSupplier);
final String authFile = authConfig.getFile();
if(authFile != null && !authFile.isEmpty()) {
final Map credentials = loadCredentials(authFile, (long) M);
ioTaskBuilder.setCredentialsMap(credentials);
} else {
final BatchSupplier secretSupplier;
final String secret = authConfig.getSecret();
if(secret == null) {
secretSupplier = null;
} else {
secretSupplier = new ConstantStringSupplier(secret);
}
ioTaskBuilder.setSecretSupplier(secretSupplier);
}
// init the items input
final String itemInputFile = inputConfig.getFile();
if(itemInput == null) {
itemInput = getItemInput(ioType, itemInputFile, itemInputPath);
if(itemInput == null) {
throw new OmgShootMyFootException("No item input available");
}
if(ItemType.DATA.equals(itemType)) {
sizeEstimate = estimateTransferSize(
(DataIoTaskBuilder) ioTaskBuilder, ioTaskBuilder.getIoType(),
(Input) itemInput
);
} else {
sizeEstimate = BUFF_SIZE_MIN;
}
}
// intercept the items input for the copy ranges support
final Range srcItemsCountRange = rangesConfig.getConcat();
if(srcItemsCountRange != null) {
if(
IoType.CREATE.equals(ioType)
&& ItemType.DATA.equals(itemType)
&& !(itemInput instanceof NewItemInput)
) {
final long srcItemsCountMin = srcItemsCountRange.getBeg();
final long srcItemsCountMax = srcItemsCountRange.getEnd();
if(srcItemsCountMin < 0) {
throw new OmgShootMyFootException(
"Source data items count min value should be more than 0"
);
}
if(srcItemsCountMax == 0 || srcItemsCountMax < srcItemsCountMin) {
throw new OmgShootMyFootException(
"Source data items count max value should be more than 0 and not less than "
+ "min value"
);
}
final List srcItemsBuff = new ArrayList<>((int) M);
final int srcItemsCount;
try {
srcItemsCount = loadSrcItems(itemInput, srcItemsBuff, (int) M);
} finally {
try {
itemInput.close();
} catch(final IOException ignored) {
}
}
if(srcItemsCount == 0) {
throw new OmgShootMyFootException(
"Available source items count " + srcItemsCount + " should be more than 0"
);
}
if(srcItemsCount < srcItemsCountMin) {
throw new OmgShootMyFootException(
"Available source items count " + srcItemsCount + " is less than configured"
+ " min " + srcItemsCountMin
);
}
if(srcItemsCount < srcItemsCountMax) {
throw new OmgShootMyFootException(
"Available source items count " + srcItemsCount + " is less than configured"
+ " max " + srcItemsCountMax
);
}
// it's safe to cast to int here because the values will not be more than
// srcItemsCount which is not more than the integer limit
((DataIoTaskBuilder) ioTaskBuilder).setSrcItemsCount(
(int) srcItemsCountMin, (int) srcItemsCountMax
);
((DataIoTaskBuilder) ioTaskBuilder).setSrcItemsForConcat(srcItemsBuff);
itemInput = getNewItemInput();
}
}
// adjust the storage drivers for the estimated transfer size
if(sizeEstimate != 0 && ItemType.DATA.equals(itemType)) {
for(final StorageDriver storageDriver : storageDrivers) {
try {
storageDriver.adjustIoBuffers(sizeEstimate, ioType);
} catch(final RemoteException e) {
LogUtil.exception(
Level.WARN, e, "Failed to adjust the storage driver buffer sizes"
);
}
}
}
final RecycleConfig recycleConfig = generatorConfig.getRecycleConfig();
final int recycleLimit = recycleConfig.getEnabled() ? recycleConfig.getLimit() : 0;
return (T) new BasicLoadGenerator<>(
itemInput, batchSize, sizeEstimate, ioTaskBuilder, countLimit, sizeLimit, recycleLimit,
shuffleFlag
);
}
private static long estimateTransferSize(
final DataIoTaskBuilder dataIoTaskBuilder, final IoType ioType,
final Input itemInput
) {
long sizeThreshold = 0;
int randomRangesCount = 0;
List fixedRanges = null;
if(dataIoTaskBuilder != null) {
sizeThreshold = dataIoTaskBuilder.getSizeThreshold();
randomRangesCount = dataIoTaskBuilder.getRandomRangesCount();
fixedRanges = dataIoTaskBuilder.getFixedRanges();
}
long itemSize = 0;
final int maxCount = 0x100;
final List items = new ArrayList<>(maxCount);
int n = 0;
try {
while(n < maxCount) {
n += itemInput.get(items, maxCount - n);
}
} catch(final EOFException ignored) {
} catch(final IOException e) {
LogUtil.exception(Level.WARN, e, "Failed to estimate the average data item size");
} finally {
try {
itemInput.reset();
} catch(final IOException e) {
LogUtil.exception(Level.WARN, e, "Failed reset the items input");
}
}
long sumSize = 0;
long minSize = Long.MAX_VALUE;
long maxSize = Long.MIN_VALUE;
long nextSize;
if(n > 0) {
try {
for(int i = 0; i < n; i++) {
nextSize = items.get(i).size();
sumSize += nextSize;
if(nextSize < minSize) {
minSize = nextSize;
}
if(nextSize > maxSize) {
maxSize = nextSize;
}
}
} catch(final IOException e) {
throw new AssertionError(e);
}
itemSize = minSize == maxSize ? sumSize / n : (minSize + maxSize) / 2;
}
switch(ioType) {
case CREATE:
return Math.min(itemSize, sizeThreshold);
case READ:
case UPDATE:
if(itemSize > 0 && randomRangesCount > 0) {
return itemSize * randomRangesCount / getRangeCount(itemSize);
} else if(fixedRanges != null && !fixedRanges.isEmpty()) {
long sizeSum = 0;
long rangeSize;
for(final Range byteRange : fixedRanges) {
rangeSize = byteRange.getSize();
if(rangeSize == -1) {
rangeSize = byteRange.getEnd() - byteRange.getBeg() + 1;
}
if(rangeSize > 0) {
sizeSum += rangeSize;
}
}
return sizeSum;
} else {
return itemSize;
}
default:
return 0;
}
}
private BatchSupplier getOutputPathSupplier()
throws OmgShootMyFootException {
final BatchSupplier pathSupplier;
String path = itemConfig.getOutputConfig().getPath();
if(path == null || path.isEmpty()) {
path = LogUtil.getDateTimeStamp();
}
if(!path.startsWith("/")) {
path = "/" + path;
}
if(-1 == path.indexOf(PATTERN_CHAR)) {
pathSupplier = new ConstantStringSupplier(path);
} else {
pathSupplier = new RangePatternDefinedSupplier(path);
}
return pathSupplier;
}
@SuppressWarnings("unchecked")
private Input getItemInput(
final IoType ioType, final String itemInputFile, final String itemInputPath
) throws OmgShootMyFootException {
if(itemInputFile == null || itemInputFile.isEmpty()) {
if(itemInputPath == null || itemInputPath.isEmpty()) {
if(IoType.CREATE.equals(ioType) || IoType.NOOP.equals(ioType)) {
itemInput = getNewItemInput();
} else {
throw new OmgShootMyFootException(
"No input (file either path) is specified for non-create generator"
);
}
} else {
final NamingConfig namingConfig = itemConfig.getNamingConfig();
final String namingPrefix = namingConfig.getPrefix();
final int namingRadix = namingConfig.getRadix();
itemInput = new StorageItemInput<>(
storageDrivers.get(0), batchSize, itemFactory, itemInputPath, namingPrefix,
namingRadix
);
}
} else {
try {
itemInput = new CsvFileItemInput<>(Paths.get(itemInputFile), itemFactory);
} catch(final NoSuchMethodException e) {
throw new RuntimeException(e);
} catch(final IOException e) {
LogUtil.exception(
Level.WARN, e, "Failed to use the item input file \"{}\"", itemInputFile
);
}
}
return itemInput;
}
private Input getNewItemInput()
throws OmgShootMyFootException {
final NamingConfig namingConfig = itemConfig.getNamingConfig();
final ItemNamingType namingType = ItemNamingType.valueOf(
namingConfig.getType().toUpperCase()
);
final String namingPrefix = namingConfig.getPrefix();
final int namingLength = namingConfig.getLength();
final int namingRadix = namingConfig.getRadix();
final long namingOffset = namingConfig.getOffset();
final ItemNameSupplier itemNameInput = new ItemNameSupplier(
namingType, namingPrefix, namingLength, namingRadix, namingOffset
);
if(itemFactory instanceof BasicDataItemFactory) {
final SizeInBytes size = itemConfig.getDataConfig().getSize();
itemInput = (Input) new NewDataItemInput(itemFactory, itemNameInput, size);
} else {
itemInput = new NewItemInput<>(itemFactory, itemNameInput);
}
return itemInput;
}
private static Map loadCredentials(final String file, final long countLimit)
throws OmgShootMyFootException {
final Map credentials = new HashMap<>();
try(final BufferedReader br = Files.newBufferedReader(Paths.get(file))) {
String line;
String parts[];
int firstCommaPos;
long count = 0;
while(null != (line = br.readLine()) && count < countLimit) {
firstCommaPos = line.indexOf(',');
if(-1 == firstCommaPos) {
Loggers.ERR.warn("Invalid credentials line: \"{}\"", line);
} else {
parts = line.split(",", 2);
credentials.put(parts[0], parts[1]);
count ++;
}
}
Loggers.MSG.info(
"Loaded {} credential pairs from the file \"{}\"", credentials.size(), file
);
} catch(final IOException e) {
LogUtil.exception(
Level.WARN, e, "Failed to load the credentials from the file \"{}\"", file
);
}
return credentials;
}
private static int loadSrcItems(
final Input itemInput, final List itemBuff, final int countLimit
) {
final LongAdder loadedCount = new LongAdder();
final ScheduledExecutorService executor = Executors.newScheduledThreadPool(
2, new LogContextThreadFactory("loadSrcItemsWorker", true)
);
final Semaphore loadFinishSemaphore = new Semaphore(1);
try {
loadFinishSemaphore.acquire();
executor.submit(
() -> {
int n = 0;
int m;
try {
while(n < countLimit) {
m = itemInput.get(itemBuff, countLimit - n);
if(m < 0) {
Loggers.MSG.info("Loaded {} items, limit reached", n);
break;
} else {
loadedCount.add(m);
n += m;
}
}
} catch(final EOFException e) {
Loggers.MSG.info("Loaded {} items, end of items input", n);
} catch(final IOException e) {
LogUtil.exception(
Level.WARN, e, "Loaded {} items, I/O failure occurred", n
);
} finally {
loadFinishSemaphore.release();
}
}
);
executor.scheduleAtFixedRate(
() -> Loggers.MSG.info("Loaded {} items from the input...", loadedCount.sum()),
0, 10, TimeUnit.SECONDS
);
loadFinishSemaphore.acquire();
} catch(final InterruptedException e) {
throw new CancellationException(e.getMessage());
} finally {
executor.shutdownNow();
}
return loadedCount.intValue();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy