org.apache.jmeter.modifiers.CounterConfig Maven / Gradle / Ivy
/*
* 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.jmeter.modifiers;
import java.io.Serializable;
import java.text.DecimalFormat;
import org.apache.jmeter.engine.event.LoopIterationEvent;
import org.apache.jmeter.engine.event.LoopIterationListener;
import org.apache.jmeter.engine.util.NoThreadClone;
import org.apache.jmeter.testelement.AbstractTestElement;
import org.apache.jmeter.testelement.ThreadListener;
import org.apache.jmeter.testelement.property.BooleanProperty;
import org.apache.jmeter.testelement.property.LongProperty;
import org.apache.jmeter.threads.JMeterContextService;
import org.apache.jmeter.threads.JMeterVariables;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Provides a counter per-thread(user) or per-thread group.
*/
public class CounterConfig extends AbstractTestElement
implements Serializable, LoopIterationListener, NoThreadClone, ThreadListener {
private static final long serialVersionUID = 235L;
private static final String START = "CounterConfig.start"; // $NON-NLS-1$
private static final String END = "CounterConfig.end"; // $NON-NLS-1$
private static final String INCREMENT = "CounterConfig.incr"; // $NON-NLS-1$
private static final String FORMAT = "CounterConfig.format"; // $NON-NLS-1$
private static final String PER_USER = "CounterConfig.per_user"; // $NON-NLS-1$
private static final String VAR_NAME = "CounterConfig.name"; // $NON-NLS-1$
private static final String RESET_ON_THREAD_GROUP_ITERATION = "CounterConfig.reset_on_tg_iteration"; // $NON-NLS-1$
private static final boolean RESET_ON_THREAD_GROUP_ITERATION_DEFAULT = false;
// This class is not cloned per thread, so this is shared
private long globalCounter = Long.MIN_VALUE;
// Used for per-thread/user numbers
private transient ThreadLocal perTheadNumber;
// Used for per-thread/user storage of increment in Thread Group Main loop
private transient ThreadLocal perTheadLastIterationNumber;
private static final Logger log = LoggerFactory.getLogger(CounterConfig.class);
private void init() { // WARNING: called from ctor so must not be overridden (i.e. must be private or final)
perTheadNumber = ThreadLocal.withInitial(this::getStart);
perTheadLastIterationNumber = ThreadLocal.withInitial(() -> Long.valueOf(1));
}
public CounterConfig() {
super();
init();
}
private Object readResolve(){
init();
return this;
}
/**
* @see LoopIterationListener#iterationStart(LoopIterationEvent)
*/
@Override
public void iterationStart(LoopIterationEvent event) {
// Cannot use getThreadContext() as not cloned per thread
JMeterVariables variables = JMeterContextService.getContext().getVariables();
long start = getStart();
long end = getEnd();
long increment = getIncrement();
if (!isPerUser()) {
synchronized (this) {
if (globalCounter == Long.MIN_VALUE || globalCounter > end) {
globalCounter = start;
}
variables.put(getVarName(), formatNumber(globalCounter));
globalCounter += increment;
}
} else {
long current = perTheadNumber.get();
if(isResetOnThreadGroupIteration()) {
int iteration = variables.getIteration();
Long lastIterationNumber = perTheadLastIterationNumber.get();
if(iteration != lastIterationNumber) {
// reset
current = getStart();
}
perTheadLastIterationNumber.set((long) iteration);
}
variables.put(getVarName(), formatNumber(current));
current += increment;
if (current > end) {
current = start;
}
perTheadNumber.set(current);
}
}
// Use format to create number; if it fails, use the default
private String formatNumber(long value){
String format = getFormat();
if (format != null && format.length() > 0) {
try {
DecimalFormat myFormatter = new DecimalFormat(format);
return myFormatter.format(value);
} catch (IllegalArgumentException ignored) {
log.warn("Error formatting {} at format {}, using default", value, format);
}
}
return Long.toString(value);
}
public void setStart(long start) {
setProperty(new LongProperty(START, start));
}
public void setStart(String start) {
setProperty(START, start);
}
public long getStart() {
return getPropertyAsLong(START);
}
public String getStartAsString() {
return getPropertyAsString(START);
}
public void setEnd(long end) {
setProperty(new LongProperty(END, end));
}
public void setEnd(String end) {
setProperty(END, end);
}
/**
* Configures if the counter must be reset on Thread Group Iteration.
* @param value boolean indicating if counter must be reset on Thread Group Iteration
*/
public void setResetOnThreadGroupIteration(boolean value) {
setProperty(RESET_ON_THREAD_GROUP_ITERATION, value, RESET_ON_THREAD_GROUP_ITERATION_DEFAULT);
}
/**
* Returns true if counter must be reset on Thread Group Iteration.
* @return true if counter must be reset on Thread Group Iteration
*/
public boolean isResetOnThreadGroupIteration() {
return getPropertyAsBoolean(RESET_ON_THREAD_GROUP_ITERATION, RESET_ON_THREAD_GROUP_ITERATION_DEFAULT);
}
/**
* Returns counter upper limit (default {@code Long.MAX_VALUE}).
* @return counter upper limit (default Long.MAX_VALUE)
*/
public long getEnd() {
long propertyAsLong = getPropertyAsLong(END);
if (propertyAsLong == 0 && "".equals(getProperty(END).getStringValue())) {
propertyAsLong = Long.MAX_VALUE;
}
return propertyAsLong;
}
public String getEndAsString(){
return getPropertyAsString(END);
}
public void setIncrement(long inc) {
setProperty(new LongProperty(INCREMENT, inc));
}
public void setIncrement(String incr) {
setProperty(INCREMENT, incr);
}
public long getIncrement() {
return getPropertyAsLong(INCREMENT);
}
public String getIncrementAsString() {
return getPropertyAsString(INCREMENT);
}
public void setIsPerUser(boolean isPer) {
setProperty(new BooleanProperty(PER_USER, isPer));
}
public boolean isPerUser() {
return getPropertyAsBoolean(PER_USER);
}
public void setVarName(String name) {
setProperty(VAR_NAME, name);
}
public String getVarName() {
return getPropertyAsString(VAR_NAME);
}
public void setFormat(String format) {
setProperty(FORMAT, format);
}
public String getFormat() {
return getPropertyAsString(FORMAT);
}
@Override
public void threadStarted() {
// nothing to do on thread start
}
@Override
public void threadFinished() {
perTheadLastIterationNumber.remove();
perTheadNumber.remove();
}
}