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

com.powsybl.metrix.tools.MappingTool Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (c) 2021, RTE (http://www.rte-france.com)
 * 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.metrix.tools;

import com.google.auto.service.AutoService;
import com.google.common.base.Stopwatch;
import com.google.common.collect.Range;
import com.powsybl.commons.datasource.DataSource;
import com.powsybl.commons.datasource.DataSourceUtil;
import com.powsybl.iidm.network.ImportConfig;
import com.powsybl.iidm.network.Network;
import com.powsybl.metrix.mapping.*;
import com.powsybl.metrix.mapping.timeseries.CalculatedTimeSeriesStore;
import com.powsybl.metrix.mapping.timeseries.InMemoryTimeSeriesStore;
import com.powsybl.timeseries.ReadOnlyTimeSeriesStore;
import com.powsybl.timeseries.ReadOnlyTimeSeriesStoreAggregator;
import com.powsybl.timeseries.TimeSeriesIndex;
import com.powsybl.timeseries.ast.NodeCalc;
import com.powsybl.tools.Command;
import com.powsybl.tools.Tool;
import com.powsybl.tools.ToolRunningContext;
import org.apache.commons.cli.CommandLine;
import org.apache.commons.cli.Option;
import org.apache.commons.cli.Options;
import org.apache.commons.io.FileSystem;
import org.codehaus.groovy.runtime.StackTraceUtils;

import java.io.IOException;
import java.io.Reader;
import java.io.UncheckedIOException;
import java.io.Writer;
import java.nio.charset.StandardCharsets;
import java.nio.file.Files;
import java.nio.file.Path;
import java.util.*;
import java.util.concurrent.TimeUnit;
import java.util.stream.Collectors;

/**
 * @author Paul Bui-Quang {@literal }
 */
@AutoService(Tool.class)
public class MappingTool implements Tool {

    private static final char SEPARATOR = ';';
    private static final String MAPPING_SYNTHESIS_DIR = "mapping-synthesis-dir";
    private static final String CHECK_EQUIPMENT_TIME_SERIES = "check-equipment-time-series";
    private static final String CHECK_VERSIONS = "check-versions";
    private static final String FIRST_VARIANT = "first-variant";
    private static final String MAX_VARIANT_COUNT = "max-variant-count";

    @Override
    public Command getCommand() {
        return new Command() {
            @Override
            public String getName() {
                return "mapping";
            }

            @Override
            public String getTheme() {
                return "Metrix";
            }

            @Override
            public String getDescription() {
                return "Time series to network mapping tool";
            }

            @Override
            public Options getOptions() {
                Options options = new Options();
                options.addOption(Option.builder()
                        .longOpt("case-file")
                        .desc("the case to which times series will be mapped")
                        .hasArg()
                        .argName("FILE")
                        .required()
                        .build());
                options.addOption(Option.builder()
                        .longOpt("mapping-file")
                        .desc("Groovy DSL file that describes the mapping")
                        .hasArg()
                        .argName("FILE")
                        .required()
                        .build());
                options.addOption(Option.builder()
                        .longOpt("time-series")
                        .desc("time series csv list")
                        .hasArg()
                        .argName("FILE1,FILE2,...")
                        .required()
                        .build());
                options.addOption(Option.builder()
                        .longOpt(MAPPING_SYNTHESIS_DIR)
                        .desc("output directory to write mapping synthesis files")
                        .hasArg()
                        .argName("DIR")
                        .build());
                options.addOption(Option.builder()
                        .longOpt(CHECK_EQUIPMENT_TIME_SERIES)
                        .desc("check equipment level time series consistency")
                        .build());
                options.addOption(Option.builder()
                        .longOpt(CHECK_VERSIONS)
                        .desc("version list to check, all if option is not set")
                        .hasArg()
                        .argName("VERSION1,VERSION2,...")
                        .build());
                options.addOption(Option.builder()
                        .longOpt("mapping-status-file")
                        .desc("check mapping status of each time series of the DB and write result to the file")
                        .hasArg()
                        .argName("FILE")
                        .build());
                options.addOption(Option.builder()
                        .longOpt("network-output-dir")
                        .desc("output directory to write IIDM networks")
                        .hasArg()
                        .argName("DIR")
                        .build());
                options.addOption(Option.builder()
                        .longOpt("equipment-time-series-dir")
                        .desc("output directory to store equipment level time series")
                        .hasArg()
                        .argName("DIR")
                        .build());
                options.addOption(Option.builder()
                        .longOpt(FIRST_VARIANT)
                        .desc("first variant to simulate")
                        .hasArg()
                        .argName("COUNT")
                        .build());
                options.addOption(Option.builder()
                        .longOpt(MAX_VARIANT_COUNT)
                        .desc("maximum number of variants simulated")
                        .hasArg()
                        .argName("COUNT")
                        .build());
                options.addOption(Option.builder()
                        .longOpt("ignore-limits")
                        .desc("ignore generator limits")
                        .build());
                options.addOption(Option.builder()
                        .longOpt("ignore-empty-filter")
                        .desc("ignore empty filter with non zero time series value")
                        .build());
                return options;
            }

            @Override
            public String getUsageFooter() {
                return null;
            }
        };
    }

    private Path getDir(CommandLine line, ToolRunningContext context, String optionName) throws IOException {
        Path dir = getFile(line, context, optionName);
        if (dir != null) {
            Files.createDirectories(dir);
        }
        return dir;
    }

    private Path getFile(CommandLine line, ToolRunningContext context, String optionName) {
        Path file = null;
        if (line.hasOption(optionName)) {
            file = context.getFileSystem().getPath(line.getOptionValue(optionName));
        }
        return file;
    }

    private Writer getWriter(Path filePath) {
        if (filePath != null) {
            try {
                return Files.newBufferedWriter(filePath, StandardCharsets.UTF_8);
            } catch (IOException e) {
                throw new UncheckedIOException(e);
            }
        }
        return null;
    }

    private ReadOnlyTimeSeriesStoreAggregator getStoreAggregator(Map nodes, ReadOnlyTimeSeriesStore store) {
        List stores = new ArrayList<>(2);
        stores.add(new CalculatedTimeSeriesStore(nodes, store));
        stores.add(store);
        return new ReadOnlyTimeSeriesStoreAggregator(stores);
    }

    @Override
    public void run(CommandLine line, ToolRunningContext context) {
        try {
            Path caseFile = context.getFileSystem().getPath(line.getOptionValue("case-file"));
            Path mappingFile = context.getFileSystem().getPath(line.getOptionValue("mapping-file"));
            List tsCsvs = Arrays.stream(line.getOptionValue("time-series").split(",")).map(String::valueOf).toList();
            if (tsCsvs.isEmpty()) {
                throw new IllegalArgumentException("Space list is empty");
            }
            Path mappingSynthesisDir = getDir(line, context, MAPPING_SYNTHESIS_DIR);
            Path mappingStatusFile = getFile(line, context, "mapping-status-file");
            boolean checkEquipmentTimeSeries = line.hasOption(CHECK_EQUIPMENT_TIME_SERIES);
            TreeSet versions = null;
            if (line.hasOption(CHECK_VERSIONS)) {
                versions = Arrays.stream(line.getOptionValue(CHECK_VERSIONS).split(",")).map(Integer::valueOf).collect(Collectors.toCollection(TreeSet::new));
                if (versions.isEmpty()) {
                    throw new IllegalArgumentException("Version list is empty");
                }
            }
            if (checkEquipmentTimeSeries && versions == null) {
                throw new IllegalArgumentException("check-versions has to be set when check-equipment-time-series is set");
            }
            int firstVariant = line.hasOption(FIRST_VARIANT) ? Integer.parseInt(line.getOptionValue(FIRST_VARIANT)) : 0;
            int maxVariantCount = line.hasOption(MAX_VARIANT_COUNT) ? Integer.parseInt(line.getOptionValue(MAX_VARIANT_COUNT)) : Integer.MAX_VALUE;

            InMemoryTimeSeriesStore store = new InMemoryTimeSeriesStore();
            store.importTimeSeries(tsCsvs.stream().map(context.getFileSystem()::getPath).toList());

            context.getOutputStream().println("Loading case...");
            Network network = Network.read(caseFile, context.getShortTimeExecutionComputationManager(), ImportConfig.load(), null);

            context.getOutputStream().println("Mapping time series to case...");
            TimeSeriesMappingConfig config;
            MappingParameters mappingParameters = MappingParameters.load();
            ComputationRange computationRange = new ComputationRange(versions != null ? versions : store.getTimeSeriesDataVersions(), firstVariant, maxVariantCount);
            TimeSeriesMappingLogger logger = new TimeSeriesMappingLogger();
            try (Reader reader = Files.newBufferedReader(mappingFile, StandardCharsets.UTF_8);
                 Writer scriptLogWriter = mappingSynthesisDir != null ? getWriter(mappingSynthesisDir.resolve("script-logs.csv")) : null) {
                TimeSeriesDslLoader dslLoader = new TimeSeriesDslLoader(reader, mappingFile.getFileName().toString());
                Stopwatch stopwatch = Stopwatch.createStarted();
                network.addListener(new NetworkTopographyChangeNotifier("extern tool", logger));
                config = dslLoader.load(network, mappingParameters, store, new DataTableStore(), scriptLogWriter, computationRange);
                context.getOutputStream().println("Mapping done in " + stopwatch.elapsed(TimeUnit.MILLISECONDS) + " ms");
            }

            if (!new TimeSeriesMappingConfigChecker(config).isMappingComplete()) {
                context.getErrorStream().println("Mapping is incomplete");
            }

            // Local parameters
            LocalParameters localParameters = new LocalParameters(config, store, network, versions);

            // Mapping Synthesis
            writeMappingSynthesis(context, localParameters, computationRange, mappingParameters, mappingSynthesisDir);

            // Mapping Status file
            writeMappingStatusFile(mappingStatusFile, context, localParameters);

            // Computing equipment time series
            writeEquipmentTimeSeries(line, context, localParameters, mappingParameters, logger);

            if (mappingSynthesisDir != null) {
                logger.writeCsv(mappingSynthesisDir.resolve("mapping-logs.csv"));
            }
        } catch (Exception e) {
            Throwable rootCause = StackTraceUtils.sanitizeRootCause(e);
            rootCause.printStackTrace(context.getErrorStream());
        }
    }

    private void writeMappingSynthesis(ToolRunningContext context, LocalParameters localParameters,
                                       ComputationRange computationRange, MappingParameters mappingParameters,
                                       Path mappingSynthesisDir) {

        TimeSeriesMappingConfigSynthesisCsvWriter csvSynthesisWriter = new TimeSeriesMappingConfigSynthesisCsvWriter(localParameters.config());
        csvSynthesisWriter.printMappingSynthesis(context.getOutputStream());

        if (mappingSynthesisDir != null) {
            context.getOutputStream().println("Writing mapping synthesis to " + mappingSynthesisDir + "...");
            csvSynthesisWriter.writeMappingSynthesis(mappingSynthesisDir);

            ReadOnlyTimeSeriesStore storeAggregator = getStoreAggregator(localParameters.config().getTimeSeriesNodes(), localParameters.store());
            TimeSeriesMappingConfigCsvWriter csvWriter = new TimeSeriesMappingConfigCsvWriter(localParameters.config(), localParameters.network(), storeAggregator, computationRange, mappingParameters.getWithTimeSeriesStats());
            csvWriter.writeMappingCsv(mappingSynthesisDir);
            csvSynthesisWriter.writeMappingSynthesisCsv(mappingSynthesisDir);
        }
    }

    private void writeMappingStatusFile(Path mappingStatusFile, ToolRunningContext context,
                                        LocalParameters localParameters) throws IOException {

        if (mappingStatusFile != null) {
            context.getOutputStream().println("Writing time series mapping status to " + mappingStatusFile + "...");
            new TimeSeriesMappingConfigStatusCsvWriter(localParameters.config(), localParameters.store()).writeTimeSeriesMappingStatus(mappingStatusFile);
        }
    }

    private void writeEquipmentTimeSeries(CommandLine line, ToolRunningContext context,
                                          LocalParameters localParameters,
                                          MappingParameters mappingParameters,
                                          TimeSeriesMappingLogger logger) throws IOException {

        Path mappingSynthesisDir = getDir(line, context, MAPPING_SYNTHESIS_DIR);
        Path equipmentTimeSeriesDir = getDir(line, context, "equipment-time-series-dir");
        Path networkOutputDir = getDir(line, context, "network-output-dir");
        boolean checkEquipmentTimeSeries = line.hasOption(CHECK_EQUIPMENT_TIME_SERIES);
        int firstVariant = line.hasOption(FIRST_VARIANT) ? Integer.parseInt(line.getOptionValue(FIRST_VARIANT)) : 0;
        int maxVariantCount = line.hasOption(MAX_VARIANT_COUNT) ? Integer.parseInt(line.getOptionValue(MAX_VARIANT_COUNT)) : Integer.MAX_VALUE;
        boolean ignoreLimits = line.hasOption("ignore-limits");
        boolean ignoreEmptyFilter = line.hasOption("ignore-empty-filter");

        // Local parameters

        if (checkEquipmentTimeSeries) {
            context.getOutputStream().println("Computing equipment time series...");
            TimeSeriesIndex index = new TimeSeriesMappingConfigTableLoader(localParameters.config(), localParameters.store()).checkIndexUnicity();

            int lastPoint = Math.min(firstVariant + maxVariantCount, index.getPointCount()) - 1;
            Range range = Range.closed(firstVariant, lastPoint);

            BalanceSummary balanceSummary = new BalanceSummary(context.getOutputStream());
            List observers = new ArrayList<>(1);
            observers.add(balanceSummary);
            if (networkOutputDir != null) {
                String cleanedNetworkId = localParameters.network().getId();
                for (char c : FileSystem.getCurrent().getIllegalFileNameChars()) {
                    cleanedNetworkId = cleanedNetworkId.replace(c, '_');
                }
                DataSource dataSource = DataSourceUtil.createDataSource(networkOutputDir.resolve(cleanedNetworkId), null);
                observers.add(new NetworkPointWriter(localParameters.network(), dataSource));
            }
            if (equipmentTimeSeriesDir != null) {
                observers.add(new EquipmentTimeSeriesWriterObserver(localParameters.network(), localParameters.config(), maxVariantCount, range, equipmentTimeSeriesDir));
                observers.add(new EquipmentGroupTimeSeriesWriterObserver(localParameters.network(), localParameters.config(), maxVariantCount, range, equipmentTimeSeriesDir));
            }

            TimeSeriesMapperParameters parameters = new TimeSeriesMapperParameters(localParameters.versions(), range, ignoreLimits,
                ignoreEmptyFilter, false, mappingParameters.getToleranceThreshold());
            TimeSeriesMapper mapper = new TimeSeriesMapper(localParameters.config(), parameters, localParameters.network(), logger);
            mapper.mapToNetwork(localParameters.store(), observers);

            if (mappingSynthesisDir != null) {
                balanceSummary.writeCsv(mappingSynthesisDir, SEPARATOR);
            }
        }
    }

    private record LocalParameters(TimeSeriesMappingConfig config, InMemoryTimeSeriesStore store, Network network,
                                   TreeSet versions) {
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy