All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.mgmtp.perfload.loadprofiles.util.PlotFileCreator Maven / Gradle / Ivy

/*
 * Copyright (c) 2014 mgm technology partners GmbH
 *
 * Licensed 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 com.mgmtp.perfload.loadprofiles.util;

import static org.apache.commons.io.FileUtils.forceMkdir;

import java.io.File;
import java.io.IOException;
import java.io.PrintWriter;
import java.text.Format;
import java.text.NumberFormat;
import java.util.Collection;
import java.util.List;

import org.apache.commons.io.IOUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import com.mgmtp.perfload.loadprofiles.generation.LoadCurveCalculator;
import com.mgmtp.perfload.loadprofiles.model.LoadCurve;
import com.mgmtp.perfload.loadprofiles.model.LoadCurveAssignment;
import com.mgmtp.perfload.loadprofiles.model.LoadEvent;

/**
 * @author mvarendo
 */
public class PlotFileCreator {
	private static final Logger log = LoggerFactory.getLogger(PlotFileCreator.class);

	/**
	 * Create a time-rate-histogram of events of the given operation. The histograms are usually
	 * used for diagnostic purposes.
	 * 
	 * @param file
	 *            The plot file
	 * @param eventList
	 *            List of events from which the histogram is derived after applying a filter on the
	 *            operation
	 * @param operationName
	 *            Name of the operation, used as filter to use only events in the histogram, which
	 *            have this operation assigned.
	 * @param nBin
	 *            Number of bins of the resulting histogram.
	 * @param xLow
	 *            time value of the left boundary of the lowest bin of the resulting histogram
	 * @param xUp
	 *            time value of the right boundary of the highest bin of the resulting histogram
	 * @param timeUnitPlot
	 *            The time unit of the resulting plot
	 */
	public static void createOperationHistogram(final File file, final Collection eventList,
			final String operationName, final int nBin, final double xLow, final double xUp, final String timeUnitPlot)
			throws IOException {
		double timeScalingFactor = LoadCurveCalculator.getTimeScalingFactor(LoadCurveCalculator.timeUnit_hour, timeUnitPlot);

		double[] xhistoLow = new double[nBin];
		double[] xhistoUp = new double[nBin];
		int[] yhisto = new int[nBin];

		double delta = nBin / (xUp - xLow);
		for (int iBin = 0; iBin < nBin; iBin++) {
			xhistoLow[iBin] = iBin / delta + xLow;
			xhistoUp[iBin] = (iBin + 1) / delta + xLow;
		}
		for (LoadEvent event : eventList) {
			if (event.getOperation().getName().equals(operationName)) {
				double time = timeScalingFactor * event.getTime();
				int xBin = (int) Math.floor((time - xLow) * delta);
				if (xBin >= 0 && xBin < nBin) {
					yhisto[xBin]++;
				} else {
					log.warn("Value " + time + " outside range [" + xLow + ", " + xUp + ")");
				}
			}
		}

		// create plot data
		double[] x = new double[nBin * 2];
		double[] y = new double[nBin * 2];

		int plotBin = 0;
		for (int iBin = 0; iBin < nBin; iBin++) {
			x[plotBin] = xhistoLow[iBin];
			y[plotBin] = yhisto[iBin] / (xhistoUp[iBin] - xhistoLow[iBin]);
			plotBin++;
			x[plotBin] = xhistoUp[iBin];
			y[plotBin] = yhisto[iBin] / (xhistoUp[iBin] - xhistoLow[iBin]);
			plotBin++;
		}
		createPlot(file, x, y, "time " + timeUnitPlot, "Histogram " + operationName);
	}

	/**
	 * Create a time-rate-histogram of events of the given client. The histograms are usually used
	 * for diagnostic purposes.
	 * 
	 * @param file
	 *            The plot file
	 * @param eventList
	 *            List of events from which the histogram is derived after applying a filter on the
	 *            operation
	 * @param clientId
	 *            Id of the client, used as filter to use only events in the histogram, which have
	 *            this clientId assigned.
	 * @param nBin
	 *            Number of bins of the resulting histogram.
	 * @param xLow
	 *            time value of the left boundary of the lowest bin of the resulting histogram
	 * @param xUp
	 *            time value of the right boundary of the highest bin of the resulting histogram
	 * @param timeUnitPlot
	 *            The time unit of the resulting plot
	 */
	public static void createClientHistogram(final File file, final Collection eventList, final int clientId,
			final int nBin, final double xLow, final double xUp, final String timeUnitPlot) throws IOException {
		double timeScalingFactor = LoadCurveCalculator.getTimeScalingFactor(LoadCurveCalculator.timeUnit_hour, timeUnitPlot);

		double[] xhistoLow = new double[nBin];
		double[] xhistoUp = new double[nBin];
		int[] yhisto = new int[nBin];

		double delta = nBin / (xUp - xLow);
		for (int iBin = 0; iBin < nBin; iBin++) {
			xhistoLow[iBin] = iBin / delta + xLow;
			xhistoUp[iBin] = (iBin + 1) / delta + xLow;
		}
		for (LoadEvent event : eventList) {
			if (event.getClientId() == clientId) {
				double time = timeScalingFactor * event.getTime();
				int xBin = (int) Math.floor((time - xLow) * delta);
				if (xBin >= 0 && xBin < nBin) {
					yhisto[xBin] += event.getOperation().getRelativeClientLoad();
				} else {
					log.warn("Value " + time + " outside range [" + xLow + ", " + xUp + ")");
				}
			}
		}

		// create plot data
		double[] x = new double[nBin * 2];
		double[] y = new double[nBin * 2];

		int plotBin = 0;
		for (int iBin = 0; iBin < nBin; iBin++) {
			x[plotBin] = xhistoLow[iBin];
			y[plotBin] = yhisto[iBin] / (xhistoUp[iBin] - xhistoLow[iBin]);
			plotBin++;
			x[plotBin] = xhistoUp[iBin];
			y[plotBin] = yhisto[iBin] / (xhistoUp[iBin] - xhistoLow[iBin]);
			plotBin++;
		}
		createPlot(file, x, y, "time " + timeUnitPlot, "Histogram Client " + clientId);
	}

	/**
	 * derive the id of an operation (position of the operation within the given array of operation
	 * names.
	 * 
	 * @param operationName
	 *            Name of operation, for which the id has to be derived.
	 * @param operationNames
	 *            Array of operation names
	 */
	private static int getOperationId(final String operationName, final String[] operationNames) {
		for (int i = 0; i < operationNames.length; i++) {
			if (operationName.equals(operationNames[i])) {
				return i;
			}
		}
		throw new IllegalArgumentException("Operation " + operationName + " not in " + operationNames);
	}

	/**
	 * Create a plot of the start times of load events for all given load curve assignements. This
	 * plot is normally used for diagnostic purposes.
	 * 
	 * @param file
	 *            The plot file
	 * @param eventList
	 *            List of events to be plotted.
	 * @param loadCurveAssignments
	 *            load curve assignements, by which the load events are grouped
	 * @param nClients
	 *            Number of clients
	 * @param timeUnitPlot
	 *            time unit of the plot.
	 */
	public static void createPlot(final File file, final Collection eventList,
			final List loadCurveAssignments, final int nClients, 
			final String timeUnitPlot) throws IOException {
		double timeScalingFactor = LoadCurveCalculator.getTimeScalingFactor(LoadCurveCalculator.timeUnit_hour, timeUnitPlot);

		int nAssignements = loadCurveAssignments.size();
		int[][] operationsOfType = new int[nAssignements][nClients];
		String[] operationNames = new String[nAssignements];
		for (int iAssignement = 0; iAssignement < nAssignements; iAssignement++) {
			operationNames[iAssignement] = loadCurveAssignments.get(iAssignement).getOperationName();
		}
		for (int iClient = 0; iClient < nClients; iClient++) {
			for (LoadEvent event : eventList) {
				if (event.getClientId() == iClient) {
					int operationId = getOperationId(event.getOperation().getName(), operationNames);
					operationsOfType[operationId][iClient]++;
				}
			}
		}
		PrintWriter pw = null;
		try {
			pw = new PrintWriter(file, "UTF-8");

			Format format = NumberFormat.getNumberInstance();
			for (int iClient = 0; iClient < nClients; iClient++) {
				for (int iAssignement = 0; iAssignement < nAssignements; iAssignement++) {
					pw.println("Eventtime " + timeUnitPlot + "; Client " + iClient + " "
							+ loadCurveAssignments.get(iAssignement).getLoadCurve().getName() + " "
							+ operationNames[iAssignement]);
					log.info("Writing " + operationsOfType[iAssignement][iClient] + " operations of type "
							+ loadCurveAssignments.get(iAssignement).getOperationName() + " for client " + iClient);
					for (LoadEvent event : eventList) {
						if (event.getClientId() == iClient) {
							if (event.getOperation().getName().equals(operationNames[iAssignement])) {
								double x = timeScalingFactor * event.getTime();
								double y = LoadCurveCalculator.r(loadCurveAssignments.get(iAssignement).getLoadCurve(),
										event.getTime());
								pw.println(format.format(x) + "; " + format.format(y));
							}
						}
					}
					pw.println();
				}
			}
		} finally {
			IOUtils.closeQuietly(pw);
		}
	}

	/**
	 * Plot the events created from a load curve. The start times of the events are used for the
	 * x-axis, the value of the load curve at this time is used as the y-axis.
	 * 
	 * @param file
	 *            The plot file
	 * @param eventList
	 *            List of events to be plotted.
	 * @param loadCurve
	 *            The load curve, from which the events are derived.
	 * @param timeUnitPlot
	 *            Time unit of the plot.
	 */
	public static void createPlot(final File file, final Collection eventList,
			final LoadCurve loadCurve, final String timeUnitPlot) throws IOException {
		double[] x = new double[eventList.size()];
		double[] y = new double[eventList.size()];

		double timeScalingFactor = LoadCurveCalculator.getTimeScalingFactor(LoadCurveCalculator.timeUnit_hour, timeUnitPlot);

		int iEvent = 0;
		for (LoadEvent event : eventList) {
			x[iEvent] = timeScalingFactor * event.getTime();
			y[iEvent++] = LoadCurveCalculator.r(loadCurve, event.getTime());
		}
		createPlot(file, x, y, "start time " + timeUnitPlot, loadCurve.getName());
	}

	/**
	 * Plot the events created from a load curve, filtered by the given client id. The start times
	 * of the events are used for the x-axis, the value of the load curve at this time is used as
	 * the y-axis.
	 * 
	 * @param file
	 *            The plot file
	 * @param eventList
	 *            List of events to be plotted.
	 * @param loadCurve
	 *            The load curve, from which the events are derived.
	 * @param clientId
	 *            The id of the client used for filtering the load events.
	 * @param timeUnitPlot
	 *            Time unit of the plot.
	 */
	public static void createPlot(final File file, final Collection eventList, final LoadCurve loadCurve,
			final int clientId, final String timeUnitPlot) throws IOException {
		double[] x = new double[eventList.size()];
		double[] y = new double[eventList.size()];

		double timeScalingFactor = LoadCurveCalculator.getTimeScalingFactor(LoadCurveCalculator.timeUnit_hour, timeUnitPlot);

		int iEvent = 0;
		for (LoadEvent event : eventList) {
			if (event.getClientId() == clientId) {
				x[iEvent] = timeScalingFactor * event.getTime();
				y[iEvent++] = LoadCurveCalculator.r(loadCurve, event.getTime());
			}
		}
		createPlot(file, x, y, "start time " + timeUnitPlot, loadCurve.getName() + " client " + clientId);
	}

	public static void createLoadCurvePlot(final File file, final LoadCurve loadCurve) throws IOException {
		double[] x = loadCurve.getTimeValues();
		double[] y = loadCurve.getRateValues();
		createPlot(file, x, y, "time " + loadCurve.getTimeUnit(), loadCurve.getName() + " " + loadCurve.getRateUnit());
	}

	/**
	 * Create a plot in .csv-format of the given x and y values, starting with a headerline
	 * containing the given xText and yText.
	 */
	private static void createPlot(final File file, final double[] x, final double[] y, final String xText, final String yText)
			throws IOException {

		PrintWriter pw = null;
		try {
			forceMkdir(file.getParentFile());
			pw = new PrintWriter(file, "UTF-8");

			Format format = NumberFormat.getNumberInstance();
			pw.println(xText + "; " + yText);
			for (int iLine = 0; iLine < x.length; iLine++) {
				pw.println(format.format(x[iLine]) + "; " + format.format(y[iLine]));
			}
		} finally {
			IOUtils.closeQuietly(pw);
		}
	}

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy