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

com.powsybl.iidm.modification.scalable.GeneratorScalable Maven / Gradle / Ivy

/**
 * Copyright (c) 2019, 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.iidm.modification.scalable;

import com.powsybl.iidm.modification.ConnectGenerator;
import com.powsybl.iidm.network.*;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import java.util.List;
import java.util.Objects;

import static com.powsybl.iidm.modification.scalable.Scalable.ScalingConvention.*;

/**
 * @author Geoffroy Jamgotchian {@literal }
 * @author Ameni Walha {@literal }
 */
class GeneratorScalable extends AbstractInjectionScalable {

    private static final Logger LOGGER = LoggerFactory.getLogger(GeneratorScalable.class);

    GeneratorScalable(String id) {
        super(id);
    }

    GeneratorScalable(String id, double minValue, double maxValue) {
        super(id, minValue, maxValue);
    }

    @Override
    public void reset(Network n) {
        Objects.requireNonNull(n);

        Generator g = n.getGenerator(id);
        if (g != null) {
            g.setTargetP(0);
        }
    }

    /**
     * {@inheritDoc}
     *
     * Default value is generator maximum power for GeneratorScalable
     */
    @Override
    public double maximumValue(Network n, ScalingConvention scalingConvention) {
        Objects.requireNonNull(n);
        Objects.requireNonNull(scalingConvention);

        Generator g = n.getGenerator(id);
        if (g != null) {
            return scalingConvention == GENERATOR ? maximumTargetP(g) : -minimumTargetP(g);
        } else {
            return 0;
        }
    }

    /**
     * {@inheritDoc}
     *
     * Default value is generator minimum power for GeneratorScalable
     */
    @Override
    public double minimumValue(Network n, ScalingConvention scalingConvention) {
        Objects.requireNonNull(n);
        Objects.requireNonNull(scalingConvention);

        Generator g = n.getGenerator(id);
        if (g != null) {
            return scalingConvention == GENERATOR ? minimumTargetP(g) : -maximumTargetP(g);
        } else {
            return 0;
        }
    }

    private double minimumTargetP(Generator gen) {
        return Math.max(gen.getMinP(), minValue);
    }

    private double maximumTargetP(Generator gen) {
        return Math.min(gen.getMaxP(), maxValue);
    }

    @Override
    public void filterInjections(Network n, List injections, List notFoundInjections) {
        Objects.requireNonNull(n);
        Objects.requireNonNull(injections);

        Generator generator = n.getGenerator(id);
        if (generator != null) {
            injections.add(generator);
        } else if (notFoundInjections != null) {
            notFoundInjections.add(id);
        }
    }

    /**
     * {@inheritDoc}
     *
     * 
    *
  • If scalingConvention is GENERATOR, the generator active power increases for positive "asked" and decreases inversely.
  • *
  • If scalingConvention is LOAD, the generator active power decreases for positive "asked" and increases inversely.
  • *
*/ @Override public double scale(Network n, double asked, ScalingParameters parameters) { Objects.requireNonNull(n); Objects.requireNonNull(parameters); if (parameters.getIgnoredInjectionIds().contains(id)) { LOGGER.info("Scaling parameters' injections to be ignored contains generator {}, discarded from scaling", id); return 0; } Generator g = n.getGenerator(id); double done = 0; if (g == null) { LOGGER.warn("Generator {} not found", id); return done; } Terminal t = g.getTerminal(); if (!t.isConnected()) { if (parameters.isReconnect()) { new ConnectGenerator(g.getId()).apply(n); LOGGER.info("Connecting {}", g.getId()); } else { LOGGER.info("Generator {} is not connected, discarded from scaling", g.getId()); return 0.; } } double oldTargetP = g.getTargetP(); double minimumTargetP = minimumTargetP(g); double maximumTargetP = maximumTargetP(g); if (!parameters.isAllowsGeneratorOutOfActivePowerLimits() && (oldTargetP < minimumTargetP || oldTargetP > maximumTargetP)) { LOGGER.error("Error scaling GeneratorScalable {}: Initial P is not in the range [Pmin, Pmax], skipped", id); return 0.; } // We use natural generator convention to compute the limits. // The actual convention is taken into account afterwards. double availableUp = maximumTargetP - oldTargetP; double availableDown = oldTargetP - minimumTargetP; if (parameters.getScalingConvention() == GENERATOR) { done = asked > 0 ? Math.min(asked, availableUp) : -Math.min(-asked, availableDown); g.setTargetP(oldTargetP + done); } else { done = asked > 0 ? Math.min(asked, availableDown) : -Math.min(-asked, availableUp); g.setTargetP(oldTargetP - done); } LOGGER.info("Change active power setpoint of {} from {} to {} (pmax={})", g.getId(), oldTargetP, g.getTargetP(), g.getMaxP()); return done; } /** * Compute the percentage of asked power available for the scale. It takes into account the scaling convention * specified by the user and the sign of the asked power. * * @param network Network on which the scaling is done * @param asked Asked power (can be positive or negative) * @param scalingPercentage Percentage of the asked power that shall be distributed to the current injection * @param scalingConvention Scaling convention (GENERATOR or LOAD) * @return the percentage of asked power available for the scale on the current injection */ double availablePowerInPercentageOfAsked(Network network, double asked, double scalingPercentage, ScalingConvention scalingConvention) { var generator = network.getGenerator(id); // In LOAD convention, a positive scale will imply a decrease of generators target power var askedPower = asked * scalingPercentage / 100; if (scalingConvention == LOAD) { askedPower = -askedPower; } if (askedPower >= 0) { var availablePower = Math.min(generator.getMaxP(), maxValue) - generator.getTargetP(); return askedPower > availablePower ? availablePower / askedPower : 100.0; } else { var availablePower = Math.max(generator.getMinP(), minValue) - generator.getTargetP(); return askedPower < availablePower ? availablePower / askedPower : 100.0; } } @Override public double getSteadyStatePower(Network network, double asked, ScalingConvention scalingConvention) { Generator generator = network.getGenerator(id); if (generator == null) { LOGGER.warn("Generator {} not found", id); return 0.0; } else { return scalingConvention == GENERATOR ? generator.getTargetP() : -generator.getTargetP(); } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy