org.apache.jmeter.functions.Groovy.java Source code

Java tutorial

Introduction

Here is the source code for org.apache.jmeter.functions.Groovy.java

Source

/*
 * 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;
    }
}