com.powsybl.computation.mpi.CsvMpiStatistics Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of powsybl-computation-mpi Show documentation
Show all versions of powsybl-computation-mpi Show documentation
A computation implementation to run computations with MPI
The newest version!
/**
* Copyright (c) 2016, All partners of the iTesla project (http://www.itesla-project.eu/consortium)
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
* SPDX-License-Identifier: MPL-2.0
*/
package com.powsybl.computation.mpi;
import com.google.common.base.Joiner;
import com.google.common.base.Joiner.MapJoiner;
import com.google.common.base.Splitter;
import com.google.common.base.Splitter.MapSplitter;
import com.powsybl.commons.PowsyblException;
import java.time.ZonedDateTime;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import java.io.*;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.time.Duration;
import java.time.format.DateTimeFormatter;
import java.util.*;
import java.util.concurrent.atomic.AtomicLong;
/**
*
* @author Geoffroy Jamgotchian {@literal }
*/
public class CsvMpiStatistics implements MpiStatistics {
private static final Logger LOGGER = LoggerFactory.getLogger(CsvMpiStatistics.class);
private static final String COMMON_FILE_TRANSFER_KEY = "COMMON_FILE_TRANSFER";
private static final String JOB_START_KEY = "JOB_START";
private static final String JOB_END_KEY = "JOB_END";
private static final String TASK_START_KEY = "TASK_START";
private static final String TASK_END_KEY = "TASK_END";
private static final String CSV_SEPARATOR = ";";
private final Path dbDir;
private final String dbName;
private final BufferedWriter internalWriter;
private final Joiner blankJoiner = Joiner.on(' ');
private final MapJoiner mapJoiner = blankJoiner.withKeyValueSeparator("=");
private static final class StatisticsReader implements AutoCloseable {
private final BufferedReader reader;
Map tasks = new HashMap<>();
Map jobs = new HashMap<>();
private static final class JobExecution {
final int jobId;
final String commandId;
final Map tags;
private JobExecution(int jobId, String commandId, Map tags) {
this.jobId = jobId;
this.commandId = commandId;
this.tags = tags;
}
}
private static final class TaskExecution {
final int taskId;
final int jobId;
final int taskIndex;
final ZonedDateTime startTime;
final int slaveRank;
final int slaveThread;
final long inputMessageSize;
Long taskDuration;
List commandsDuration;
Long dataTransferDuration;
Long outputMessageSize;
Long workingDataSize;
Integer exitCode;
private TaskExecution(int taskId, int jobId, int taskIndex, ZonedDateTime startTime, int slaveRank, int slaveThread, long inputMessageSize) {
this.taskId = taskId;
this.jobId = jobId;
this.taskIndex = taskIndex;
this.startTime = startTime;
this.slaveRank = slaveRank;
this.slaveThread = slaveThread;
this.inputMessageSize = inputMessageSize;
}
}
private static final class CommonFileTransfer {
String fileName;
long size;
long duration;
private CommonFileTransfer(String fileName, long size, long duration) {
this.fileName = fileName;
this.size = size;
this.duration = duration;
}
}
private interface Handler {
void onTaskEnd(TaskExecution task, JobExecution job);
void onCommonFileTransfer(CommonFileTransfer commonFileTransfer);
}
private abstract static class AbstractHandler implements Handler {
@Override
public void onTaskEnd(TaskExecution task, JobExecution job) {
}
@Override
public void onCommonFileTransfer(CommonFileTransfer commonFileTransfer) {
}
}
private StatisticsReader(Path tasksCsv) throws IOException {
reader = Files.newBufferedReader(tasksCsv, StandardCharsets.UTF_8);
}
private void read(Handler handler) throws IOException {
Splitter blankSplitter = Splitter.on(' ');
MapSplitter mapSplitter = blankSplitter.withKeyValueSeparator('=');
String line;
int jobId;
int taskId;
while ((line = reader.readLine()) != null) {
if (line.trim().isEmpty()) {
continue;
}
String[] tokens = line.split(CSV_SEPARATOR);
if (tokens.length < 1) {
throw new PowsyblException("Cannot detect line key");
}
String key = tokens[0];
switch (key) {
case COMMON_FILE_TRANSFER_KEY:
checkTokenSize(4, tokens.length, line, key);
String fileName = tokens[1];
long size = Long.parseLong(tokens[2]);
long duration = Long.parseLong(tokens[3]);
handler.onCommonFileTransfer(new CommonFileTransfer(fileName, size, duration));
break;
case JOB_START_KEY:
checkTokenSize(3, 4, tokens.length, line, key);
jobId = Integer.parseInt(tokens[1]);
String commandId = tokens[2];
Map tags = null;
if (tokens.length == 4) {
tags = mapSplitter.split(tokens[3]);
}
jobs.put(jobId, new JobExecution(jobId, commandId, tags));
break;
case JOB_END_KEY:
checkTokenSize(2, tokens.length, line, key);
jobId = Integer.parseInt(tokens[1]);
jobs.remove(jobId);
break;
case TASK_START_KEY:
checkTokenSize(8, tokens.length, line, key);
taskId = Integer.parseInt(tokens[1]);
jobId = Integer.parseInt(tokens[2]);
int taskIndex = Integer.parseInt(tokens[3]);
ZonedDateTime startTime = ZonedDateTime.parse(tokens[4]);
int slaveRank = Integer.parseInt(tokens[5]);
int slaveThread = Integer.parseInt(tokens[6]);
long inputMessageSize = Long.parseLong(tokens[7]);
tasks.put(taskId, new TaskExecution(taskId, jobId, taskIndex, startTime, slaveRank, slaveThread, inputMessageSize));
break;
case TASK_END_KEY:
checkTokenSize(8, tokens.length, line, key);
taskId = Integer.parseInt(tokens[1]);
TaskExecution task = tasks.get(taskId);
task.taskDuration = Long.parseLong(tokens[2]);
task.commandsDuration = new ArrayList<>();
for (String s : blankSplitter.split(tokens[3])) {
task.commandsDuration.add(Long.parseLong(s));
}
task.dataTransferDuration = Long.parseLong(tokens[4]);
task.outputMessageSize = Long.parseLong(tokens[5]);
task.workingDataSize = Long.parseLong(tokens[6]);
task.exitCode = Integer.parseInt(tokens[7]);
JobExecution job = jobs.get(task.jobId);
handler.onTaskEnd(task, job);
tasks.remove(taskId);
break;
default:
throw new PowsyblException("Unknown key " + key);
}
}
}
private static PowsyblException createIncorrectLineException(String key, String line) {
return new PowsyblException("Incorrect " + key + " line '" + line + "'");
}
private static void checkTokenSize(int expected, int actual, String line, String key) {
if (actual != expected) {
throw createIncorrectLineException(key, line);
}
}
private static void checkTokenSize(int expected1, int expected2, int actual, String line, String key) {
if (actual != expected1 && actual != expected2) {
throw createIncorrectLineException(key, line);
}
}
@Override
public void close() throws IOException {
reader.close();
}
}
public CsvMpiStatistics(Path dbDir, String dbName) throws IOException {
Objects.requireNonNull(dbDir);
Objects.requireNonNull(dbName);
this.dbDir = dbDir;
this.dbName = dbName;
Path csvFile = dbDir.resolve(dbName + ".csv");
// if file already exists, create a backup
if (Files.exists(csvFile)) {
int i = 1;
Path savCsvFile;
do {
savCsvFile = csvFile.getParent().resolve(csvFile.getFileName() + "." + i++);
} while (Files.exists(savCsvFile));
Files.move(csvFile, savCsvFile);
}
internalWriter = Files.newBufferedWriter(csvFile, StandardCharsets.UTF_8);
}
@Override
public void logCommonFileTransfer(String fileName, int chunk, long size, long duration) {
try {
internalWriter.write(COMMON_FILE_TRANSFER_KEY);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(fileName);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(size));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(duration));
internalWriter.newLine();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
@Override
public void logJobStart(int jobId, String commandId, Map tags) {
try {
internalWriter.write(JOB_START_KEY);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(jobId));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(commandId);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(tags != null ? mapJoiner.join(tags) : "");
internalWriter.newLine();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
@Override
public void logJobEnd(int jobId) {
try {
internalWriter.write(JOB_END_KEY);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(jobId));
internalWriter.newLine();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
@Override
public void logTaskStart(int taskId, int jobId, int taskIndex, ZonedDateTime startTime, int slaveRank, int slaveThread, long inputMessageSize) {
try {
internalWriter.write(TASK_START_KEY);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(taskId));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(jobId));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(taskIndex));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(startTime.toString());
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(slaveRank));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(slaveThread));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(inputMessageSize));
internalWriter.newLine();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
@Override
public void logTaskEnd(int taskId, long taskDuration, List commandsDuration, long dataTransferDuration, long outputMessageSize, long workingDataSize, int exitCode) {
try {
internalWriter.write(TASK_END_KEY);
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(taskId));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(taskDuration));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(blankJoiner.join(commandsDuration));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(dataTransferDuration));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(outputMessageSize));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Long.toString(workingDataSize));
internalWriter.write(CSV_SEPARATOR);
internalWriter.write(Integer.toString(exitCode));
internalWriter.newLine();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
@Override
public void exportTasksToCsv(final Writer writer) {
try {
Path csv = dbDir.resolve(dbName + ".csv");
writer.write("Task Id " + CSV_SEPARATOR +
"Job Id" + CSV_SEPARATOR +
"Task index" + CSV_SEPARATOR +
"Command Id" + CSV_SEPARATOR +
"Tags" + CSV_SEPARATOR +
"Start time" + CSV_SEPARATOR +
"Slave rank" + CSV_SEPARATOR +
"Slave thread" + CSV_SEPARATOR +
"Input message size (bytes)" + CSV_SEPARATOR +
"Task duration (ms)" + CSV_SEPARATOR +
"Commands duration (ms)" + CSV_SEPARATOR +
"Data transfer duration (ms)" + CSV_SEPARATOR +
"Output message size (bytes)" + CSV_SEPARATOR +
"Working data size (bytes)" + CSV_SEPARATOR +
"Exit code");
writer.write("\n");
try (StatisticsReader reader = new StatisticsReader(csv)) {
reader.read(new StatisticsReader.AbstractHandler() {
@Override
public void onTaskEnd(StatisticsReader.TaskExecution task, StatisticsReader.JobExecution job) {
try {
String taskDuration = Objects.toString(task.taskDuration, "");
String commandsDuration = "";
if (task.commandsDuration != null) {
commandsDuration = blankJoiner.join(task.commandsDuration);
}
String dataTransferDuration = Objects.toString(task.dataTransferDuration, "");
String outputMessageSize = Objects.toString(task.outputMessageSize, "");
String workingDataSize = Objects.toString(task.workingDataSize, "");
String exitCode = Objects.toString(task.exitCode, "");
writer.write(task.taskId + CSV_SEPARATOR +
task.jobId + CSV_SEPARATOR +
task.taskIndex + CSV_SEPARATOR +
job.commandId + CSV_SEPARATOR +
(job.tags != null ? mapJoiner.join(job.tags) : "") + CSV_SEPARATOR +
task.startTime.format(DateTimeFormatter.ofPattern("dd/MM/yyyy HH:mm:ss")) + CSV_SEPARATOR +
task.slaveRank + CSV_SEPARATOR +
task.slaveThread + CSV_SEPARATOR +
task.inputMessageSize + CSV_SEPARATOR +
taskDuration + CSV_SEPARATOR +
commandsDuration + CSV_SEPARATOR +
dataTransferDuration + CSV_SEPARATOR +
outputMessageSize + CSV_SEPARATOR +
workingDataSize + CSV_SEPARATOR +
exitCode);
writer.write("\n");
} catch (IOException e) {
LOGGER.error(e.toString(), e);
}
}
});
}
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
public static void exportCommonFileTransferDuration(Path dbDir, String dbName) throws IOException {
Objects.requireNonNull(dbDir);
Objects.requireNonNull(dbName);
Path csv = dbDir.resolve(dbName + ".csv");
Path commonFilesTransferCsv = dbDir.resolve("common-files-transfer.csv");
logWritingPath(commonFilesTransferCsv);
try (BufferedWriter commonFilesTransferWriter = Files.newBufferedWriter(commonFilesTransferCsv, StandardCharsets.UTF_8)) {
commonFilesTransferWriter.write("File name" + CSV_SEPARATOR + "File size (bytes)" + CSV_SEPARATOR + "Transfer duration (ms)");
commonFilesTransferWriter.newLine();
try (StatisticsReader reader = new StatisticsReader(csv)) {
reader.read(new StatisticsReader.AbstractHandler() {
@Override
public void onCommonFileTransfer(StatisticsReader.CommonFileTransfer transfer) {
try {
commonFilesTransferWriter.write(transfer.fileName + CSV_SEPARATOR + transfer.size + CSV_SEPARATOR + transfer.duration);
commonFilesTransferWriter.newLine();
} catch (IOException e) {
LOGGER.error(e.toString(), e);
}
}
});
}
}
}
private static void logWritingPath(Path path) {
LOGGER.info("Writing {}", path);
}
public static void exportTaskCount(Path dbDir, String dbName) throws IOException {
Objects.requireNonNull(dbDir);
Objects.requireNonNull(dbName);
Path csv = dbDir.resolve(dbName + ".csv");
class CommandStats {
int count = 0;
int ok = 0;
}
Path tasksCountCsv = dbDir.resolve("tasks-count.csv");
logWritingPath(tasksCountCsv);
final Map tasksPerCommandId = new HashMap<>();
try (StatisticsReader reader = new StatisticsReader(csv)) {
reader.read(new StatisticsReader.AbstractHandler() {
@Override
public void onTaskEnd(StatisticsReader.TaskExecution task, StatisticsReader.JobExecution job) {
CommandStats stats = tasksPerCommandId.get(job.commandId);
if (stats == null) {
stats = new CommandStats();
tasksPerCommandId.put(job.commandId, stats);
}
stats.count++;
if (task.exitCode == 0) {
stats.ok++;
}
}
});
}
try (BufferedWriter writer = Files.newBufferedWriter(tasksCountCsv, StandardCharsets.UTF_8)) {
writer.write("Command Id" + CSV_SEPARATOR + "Executions" + CSV_SEPARATOR + "OK rate");
writer.newLine();
for (Map.Entry entry : tasksPerCommandId.entrySet()) {
String commandId = entry.getKey();
CommandStats stats = entry.getValue();
writer.write(commandId + CSV_SEPARATOR + stats.count + CSV_SEPARATOR + ((float) stats.ok / stats.count));
writer.newLine();
}
}
}
public static void exportBusyCores(Path dbDir, String dbName) throws IOException {
Objects.requireNonNull(dbDir);
Objects.requireNonNull(dbName);
Path csv = dbDir.resolve(dbName + ".csv");
Path busyCoresCsv = dbDir.resolve("busy-cores.csv");
logWritingPath(busyCoresCsv);
final ZonedDateTime[] min = new ZonedDateTime[1];
final ZonedDateTime[] max = new ZonedDateTime[1];
try (StatisticsReader reader = new StatisticsReader(csv)) {
reader.read(new StatisticsReader.AbstractHandler() {
@Override
public void onTaskEnd(StatisticsReader.TaskExecution task, StatisticsReader.JobExecution job) {
if (min[0] == null || task.startTime.compareTo(min[0]) < 0) {
min[0] = task.startTime;
}
if (task.taskDuration != null) {
ZonedDateTime endTime = task.startTime.plusNanos((int) (long) (task.taskDuration * 1e6));
if (max[0] == null || endTime.compareTo(max[0]) > 0) {
max[0] = endTime;
}
}
}
});
}
final int secs = (int) Duration.between(min[0], max[0]).getSeconds() + 1;
final int[] busyCores = new int[secs];
Arrays.fill(busyCores, 0);
try (StatisticsReader reader = new StatisticsReader(csv)) {
reader.read(new StatisticsReader.AbstractHandler() {
@Override
public void onTaskEnd(StatisticsReader.TaskExecution task, StatisticsReader.JobExecution job) {
int s1 = (int) Duration.between(min[0], task.startTime).getSeconds();
int s2;
if (task.taskDuration != null) {
s2 = s1 + (int) ((float) task.taskDuration / 1000);
} else {
s2 = secs - 1;
}
for (int s = s1; s <= s2; s++) {
busyCores[s]++;
}
}
});
}
try (BufferedWriter writer = Files.newBufferedWriter(busyCoresCsv, StandardCharsets.UTF_8)) {
writer.write("Second" + CSV_SEPARATOR + "Busy cores");
writer.newLine();
for (int s = 0; s < busyCores.length; s++) {
writer.write(Integer.toString(s));
writer.write(CSV_SEPARATOR);
writer.write(Integer.toString(busyCores[s]));
writer.newLine();
}
}
}
public static void exportWorkingDataSize(Path dbDir, String dbName) throws IOException {
Objects.requireNonNull(dbDir);
Objects.requireNonNull(dbName);
Path csv = dbDir.resolve(dbName + ".csv");
Path workingDataSizeCsv = dbDir.resolve("working-data-size.csv");
logWritingPath(workingDataSizeCsv);
final Map workingDataSizePerSlave = new HashMap<>();
final long[] totalWorkingDataSize = new long[1];
try (StatisticsReader reader = new StatisticsReader(csv)) {
reader.read(new StatisticsReader.AbstractHandler() {
@Override
public void onTaskEnd(StatisticsReader.TaskExecution task, StatisticsReader.JobExecution job) {
if (task.workingDataSize != null) {
String slaveId = task.slaveRank + "_" + task.slaveThread;
AtomicLong workingDataSize = workingDataSizePerSlave.computeIfAbsent(slaveId, k -> new AtomicLong());
workingDataSize.addAndGet(task.workingDataSize);
totalWorkingDataSize[0] += task.workingDataSize;
}
}
});
}
try (BufferedWriter writer = Files.newBufferedWriter(workingDataSizeCsv, StandardCharsets.UTF_8)) {
writer.write("Slave" + CSV_SEPARATOR + "Working data size (bytes)");
writer.newLine();
for (Map.Entry entry : workingDataSizePerSlave.entrySet()) {
String slaveId = entry.getKey();
long workingDataSize = entry.getValue().get();
writer.write(slaveId + CSV_SEPARATOR + workingDataSize);
writer.newLine();
}
writer.write("Total" + CSV_SEPARATOR + totalWorkingDataSize[0]);
writer.newLine();
}
}
@Override
public void close() {
try {
internalWriter.close();
} catch (IOException e) {
throw new UncheckedIOException(e);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy