blob: c942ee45eacbb47384aea7f291c5ae2db47b266b [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.report.dashboard;
import java.io.File;
import java.io.FileFilter;
import java.io.IOException;
import java.io.Writer;
import java.nio.file.Files;
import java.util.HashMap;
import java.util.Map;
import org.apache.commons.io.FileUtils;
import org.apache.commons.lang3.StringUtils;
import org.apache.jmeter.JMeter;
import org.apache.jmeter.report.config.ExporterConfiguration;
import org.apache.jmeter.report.config.ReportGeneratorConfiguration;
import org.apache.jmeter.report.processor.ListResultData;
import org.apache.jmeter.report.processor.MapResultData;
import org.apache.jmeter.report.processor.SampleContext;
import org.apache.jmeter.report.processor.ValueResultData;
import org.apache.jmeter.util.JMeterUtils;
import org.apache.jorphan.util.JOrphanUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.fasterxml.jackson.core.StreamReadFeature;
import com.fasterxml.jackson.databind.ObjectWriter;
import com.fasterxml.jackson.databind.json.JsonMapper;
/**
* Implementation of {@link DataExporter} that exports statistics to JSON
*
* @since 5.1
*/
public class JsonExporter extends AbstractDataExporter {
private static final Logger LOGGER = LoggerFactory.getLogger(JsonExporter.class);
public static final String OUTPUT_FILENAME = "statistics.json";
private static final FileFilter JSON_FILE_FILTER =
file -> file.isFile() && file.getName().equals(OUTPUT_FILENAME);
private final static ObjectWriter OBJECT_WRITER = JsonMapper.builder()
// See https://github.com/FasterXML/jackson-core/issues/991
.enable(StreamReadFeature.INCLUDE_SOURCE_IN_LOCATION)
.build()
.writerWithDefaultPrettyPrinter();
public JsonExporter() {
super();
}
@Override
public void export(SampleContext context, File file, ReportGeneratorConfiguration reportGeneratorConfiguration)
throws ExportException {
Object data = context.getData().get(ReportGenerator.STATISTICS_SUMMARY_CONSUMER_NAME);
if (data instanceof MapResultData) {
LOGGER.info("Found data for consumer {} in context", ReportGenerator.STATISTICS_SUMMARY_CONSUMER_NAME);
MapResultData result = (MapResultData) data;
Map<String, SamplingStatistic> statistics = new HashMap<>();
MapResultData overallData = (MapResultData) result.getResult("overall");
LOGGER.info("Creating statistics for overall");
createStatistic(statistics, overallData);
ListResultData itemsData = (ListResultData) result.getResult("items");
LOGGER.info("Creating statistics for other transactions");
itemsData.forEach(r -> createStatistic(statistics, (MapResultData)r));
LOGGER.info("Checking output folder");
File outputDir = checkAndGetOutputFolder(reportGeneratorConfiguration);
File outputFile = new File(outputDir, OUTPUT_FILENAME);
LOGGER.info("Writing statistics JSON to {}", outputFile);
try (Writer fileWriter = Files.newBufferedWriter(outputFile.toPath())) {
OBJECT_WRITER.writeValue(fileWriter, statistics);
} catch (IOException e) {
throw new ExportException("Error generating JSON statistics file to " + outputFile +" for "+statistics, e);
}
}
}
/**
* Check folder and return output folder.
* @param reportGeneratorConfiguration {@link ReportGeneratorConfiguration}
* @return {@link File} output folder
* @throws ExportException
*/
private File checkAndGetOutputFolder(ReportGeneratorConfiguration reportGeneratorConfiguration)
throws ExportException {
final ExporterConfiguration exportCfg = reportGeneratorConfiguration
.getExportConfigurations().get(getName());
// Get output directory property value
File outputDir = getPropertyFromConfig(exportCfg, HtmlTemplateExporter.OUTPUT_DIR,
new File(JMeterUtils.getJMeterBinDir(), HtmlTemplateExporter.OUTPUT_DIR_NAME_DEFAULT), File.class);
String globallyDefinedOutputDir = JMeterUtils.getProperty(JMeter.JMETER_REPORT_OUTPUT_DIR_PROPERTY);
if(!StringUtils.isEmpty(globallyDefinedOutputDir)) {
outputDir = new File(globallyDefinedOutputDir);
}
JOrphanUtils.canSafelyWriteToFolder(outputDir, JSON_FILE_FILTER);
try {
FileUtils.forceMkdir(outputDir);
} catch (IOException ex) {
throw new ExportException("Error creating output folder "+outputDir.getAbsolutePath(), ex);
}
return outputDir;
}
private static void createStatistic(Map<? super String, ? super SamplingStatistic> statistics, MapResultData resultData) {
LOGGER.debug("Creating statistics for result data:{}", resultData);
SamplingStatistic statistic = new SamplingStatistic();
ListResultData listResultData = (ListResultData) resultData.getResult("data");
statistic.setTransaction((String) ((ValueResultData)listResultData.get(0)).getValue());
statistic.setSampleCount((Long) ((ValueResultData)listResultData.get(1)).getValue());
statistic.setErrorCount((Long) ((ValueResultData)listResultData.get(2)).getValue());
statistic.setErrorPct(((Double) ((ValueResultData)listResultData.get(3)).getValue()).floatValue());
statistic.setMeanResTime((Double) ((ValueResultData)listResultData.get(4)).getValue());
statistic.setMinResTime((Long) ((ValueResultData)listResultData.get(5)).getValue());
statistic.setMaxResTime((Long) ((ValueResultData)listResultData.get(6)).getValue());
statistic.setMedianResTime((Double) ((ValueResultData)listResultData.get(7)).getValue());
statistic.setPct1ResTime((Double) ((ValueResultData)listResultData.get(8)).getValue());
statistic.setPct2ResTime((Double) ((ValueResultData)listResultData.get(9)).getValue());
statistic.setPct3ResTime((Double) ((ValueResultData)listResultData.get(10)).getValue());
statistic.setThroughput((Double) ((ValueResultData)listResultData.get(11)).getValue());
statistic.setReceivedKBytesPerSec((Double) ((ValueResultData)listResultData.get(12)).getValue());
statistic.setSentKBytesPerSec((Double) ((ValueResultData)listResultData.get(13)).getValue());
statistics.put(statistic.getTransaction(), statistic);
}
}