blob: 874ccc97c16abe07e8aa089f752defd92d26bef6 [file] [log] [blame]
/*
* 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.functions;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileReader;
import java.util.Collection;
import java.util.LinkedList;
import java.util.List;
import java.util.Properties;
import javax.script.Bindings;
import javax.script.ScriptEngine;
import org.apache.commons.lang3.StringUtils;
import org.apache.jmeter.engine.util.CompoundVariable;
import org.apache.jmeter.samplers.SampleResult;
import org.apache.jmeter.samplers.Sampler;
import org.apache.jmeter.threads.JMeterContext;
import org.apache.jmeter.threads.JMeterContextService;
import org.apache.jmeter.threads.JMeterVariables;
import org.apache.jmeter.util.JMeterUtils;
import org.apache.jmeter.util.JSR223TestElement;
import org.apache.jorphan.logging.LoggingManager;
import org.apache.log.Logger;
/**
* __groovy function
* Provides a Groovy interpreter
* @since 3.1
*/
public class Groovy extends AbstractFunction {
private static final Logger log = LoggingManager.getLoggerForClass();
private static final String GROOVY_ENGINE_NAME = "groovy";
private static final List<String> DESCRIPTION = new LinkedList<>();
private static final String KEY = "__groovy"; //$NON-NLS-1$
public static final String INIT_FILE = "groovy.utilities"; //$NON-NLS-1$
static {
DESCRIPTION.add(JMeterUtils.getResString("groovy_function_expression"));// $NON-NLS1$
DESCRIPTION.add(JMeterUtils.getResString("function_name_paropt"));// $NON-NLS1$
}
private Object[] values;
private ScriptEngine scriptEngine;
public Groovy() {
}
/**
* Populate variables to be passed to scripts
* @param bindings Bindings
*/
protected void populateBindings(Bindings bindings) {
}
/** {@inheritDoc} */
@Override
public synchronized String execute(SampleResult previousResult, Sampler currentSampler)
throws InvalidVariableException {
Bindings bindings = scriptEngine.createBindings();
populateBindings(bindings);
String script = ((CompoundVariable) values[0]).execute();
String varName = ""; //$NON-NLS-1$
if (values.length > 1) {
varName = ((CompoundVariable) values[1]).execute().trim();
}
String resultStr = ""; //$NON-NLS-1$
try {
// Pass in some variables
if (currentSampler != null) {
bindings.put("sampler", currentSampler); // $NON-NLS-1$
}
if (previousResult != null) {
bindings.put("prev", previousResult); //$NON-NLS-1$
}
bindings.put("log", log); // $NON-NLS-1$ (this name is fixed)
// Add variables for access to context and variables
bindings.put("threadName", Thread.currentThread().getName());
JMeterContext jmctx = JMeterContextService.getContext();
bindings.put("ctx", jmctx); // $NON-NLS-1$ (this name is fixed)
JMeterVariables vars = jmctx.getVariables();
bindings.put("vars", vars); // $NON-NLS-1$ (this name is fixed)
Properties props = JMeterUtils.getJMeterProperties();
bindings.put("props", props); // $NON-NLS-1$ (this name is fixed)
// For use in debugging:
bindings.put("OUT", System.out); // $NON-NLS-1$ (this name is fixed)
// Execute the script
Object out = scriptEngine.eval(script, bindings);
if (out != null) {
resultStr = out.toString();
}
if (varName.length() > 0) {// vars will be null on TestPlan
if(vars != null) {
vars.put(varName, resultStr);
}
}
} catch (Exception ex) // Mainly for bsh.EvalError
{
log.warn("Error running groovy script", ex);
}
if(log.isDebugEnabled()) {
log.debug("__groovy("+script+","+varName+")=" + resultStr);
}
return resultStr;
}
/*
* Helper method for use by scripts
*
*/
public void log_info(String s) {
log.info(s);
}
/** {@inheritDoc} */
@Override
public void setParameters(Collection<CompoundVariable> parameters) throws InvalidVariableException {
checkParameterCount(parameters, 1, 2);
values = parameters.toArray();
scriptEngine = JSR223TestElement.getInstance().getEngineByName(GROOVY_ENGINE_NAME); //$NON-NLS-N$
String fileName = JMeterUtils.getProperty(INIT_FILE);
if(!StringUtils.isEmpty(fileName)) {
File file = new File(fileName);
if(!(file.exists() && file.canRead())) {
// File maybe relative to JMeter home
File oldFile = file;
file = new File(JMeterUtils.getJMeterHome(), fileName);
if(!(file.exists() && file.canRead())) {
throw new InvalidVariableException("Cannot read file, neither from:"+oldFile.getAbsolutePath()+
", nor from:"+file.getAbsolutePath()+", check property '"+INIT_FILE+"'");
}
}
try (FileReader fr = new FileReader(file); BufferedReader reader = new BufferedReader(fr)) {
Bindings bindings = scriptEngine.createBindings();
bindings.put("log", log);
scriptEngine.eval(reader, bindings);
} catch(Exception ex) {
throw new InvalidVariableException("Failed loading script:"+file.getAbsolutePath(), ex);
}
}
}
/** {@inheritDoc} */
@Override
public String getReferenceKey() {
return KEY;
}
/** {@inheritDoc} */
@Override
public List<String> getArgumentDesc() {
return DESCRIPTION;
}
}