Source code for gemseo.core.coupling_structure

# -*- coding: utf-8 -*-
# Copyright 2021 IRT Saint Exupéry, https://www.irt-saintexupery.com
#
# This program is free software; you can redistribute it and/or
# modify it under the terms of the GNU Lesser General Public
# License version 3 as published by the Free Software Foundation.
#
# This program is distributed in the hope that it will be useful,
# but WITHOUT ANY WARRANTY; without even the implied warranty of
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
# Lesser General Public License for more details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program; if not, write to the Free Software Foundation,
# Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.

# Contributors:
#    INITIAL AUTHORS - initial API and implementation and/or initial
#                         documentation
#        :author: Charlie Vanaret
#    OTHER AUTHORS   - MACROSCOPIC CHANGES
#          Arthur Piat: greatly improve the N2 layout
"""Graph-based analysis of the weak and strong couplings between several disciplines."""

from __future__ import unicode_literals

import itertools
import logging
from typing import Dict, List, Sequence, Set, Tuple, Union

import matplotlib.pyplot as plt
from matplotlib.figure import Figure
from matplotlib.text import Text
from numpy import array
from pylab import gca

from gemseo.core.dependency_graph import DependencyGraph
from gemseo.core.discipline import MDODiscipline
from gemseo.utils.n2d3.n2_html import N2HTML
from gemseo.utils.py23_compat import Path, string_types

LOGGER = logging.getLogger(__name__)

NodeType = Tuple[List[str], List[str]]
EdgesType = Dict[int, Dict[int, List[str]]]
GraphType = Dict[int, Set[int]]
ComponentType = Tuple[int]


[docs]class MDOCouplingStructure(object): """Structure of the couplings between several disciplines. The methods of this class include the computation of weak, strong or all couplings. Attributes: disciplines (Sequence[MDODiscipline]): The disciplines. graph (DependencyGraph): The directed graph of the disciplines. sequence (List[List[Tuple[MDODiscipline]]]): The sequence of execution of the disciplines. """ def __init__( self, disciplines, # type: Sequence[MDODiscipline] ): # type: (...) -> None """ Args: disciplines: The disciplines that possibly exchange coupling variables. """ self.disciplines = disciplines self.graph = DependencyGraph(disciplines) self.sequence = self.graph.get_execution_sequence()
[docs] @staticmethod def is_self_coupled( discipline, # type: MDODiscipline ): # type: (...) -> bool """Test if the discipline is self-coupled. Self-coupling means that one of its outputs is also an input. Args: discipline: The discipline. Returns: Whether the discipline is self-coupled. """ return ( len( set(discipline.get_input_data_names()) & set(discipline.get_output_data_names()) ) > 0 )
# methods that determine strong/weak/all couplings
[docs] def strongly_coupled_disciplines( self, add_self_coupled=True, # type: bool by_group=False, # type: bool ): # type: (...) -> Union[List[MDODiscipline],List[List[MDODiscipline]]] """Determines the strongly coupled disciplines, that is the disciplines that occur in (possibly different) MDAs. Args: add_self_coupled: if True, adds the disciplines that are self-coupled to the list of strongly coupled disciplines by_group: if True, returns a list of list of strongly coupled diciplines where the sublists contains the groups of disciplines that are strongly coupled together. if False, returns a single list Returns: The coupled disciplines list or list of list """ strong_disciplines = [] for parallel_tasks in self.sequence: for component in parallel_tasks: # find MDAs if len(component) > 1: if by_group: strong_disciplines.append(component) else: strong_disciplines += component elif add_self_coupled: for discipline in component: if self.is_self_coupled(discipline): if by_group: strong_disciplines.append([discipline]) else: strong_disciplines.append(discipline) return strong_disciplines
[docs] def weakly_coupled_disciplines(self): # type: (...) -> List[MDODiscipline] """Determine the weakly coupled disciplines. These are the disciplines that do not occur in MDAs. Returns: The weakly coupled disciplines. """ weak_disciplines = [] for parallel_tasks in self.sequence: for component in parallel_tasks: # find single disciplines if len(component) == 1 and not self.is_self_coupled(component[0]): weak_disciplines.append(component[0]) return weak_disciplines
[docs] def strong_couplings(self): # type: (...) -> List[str] """Determine the strong couplings. These are the outputs of the strongly coupled disciplines that are also inputs of the strongly coupled disciplines. Returns: The names of the strongly coupling variables. """ strong_disciplines = self.strongly_coupled_disciplines( add_self_coupled=True, by_group=True ) # determine strong couplings = the outputs of the strongly coupled # disciplines that are inputs of any other discipline strong_couplings = set() for group in strong_disciplines: inputs = itertools.chain(*[disc.get_input_data_names() for disc in group]) outputs = itertools.chain(*[disc.get_output_data_names() for disc in group]) strong_couplings.update(set(inputs) & set(outputs)) return sorted(list(strong_couplings))
[docs] def weak_couplings(self): # type: (...) -> List[str] """Determine the weak couplings. These are the outputs of the weakly coupled disciplines. Returns: The names of the weakly coupling variables. """ weak_disciplines = self.weakly_coupled_disciplines() # determine strong couplings = the outputs of the weakly coupled # disciplines that are inputs of any other discipline weak_couplings = set() for weak_discipline in weak_disciplines: weak_couplings.update(weak_discipline.get_output_data_names()) return sorted(list(weak_couplings))
[docs] def input_couplings( self, discipline, # type: MDODiscipline ): # type: (...) -> List[str] """Compute all the input coupling variables of a discipline. Args: discipline: The discipline. Returns: The names of the input coupling variables. """ input_names = discipline.get_input_data_names() strong_couplings = self.strong_couplings() return sorted([name for name in input_names if name in strong_couplings])
[docs] def get_all_couplings(self): # type: (...) -> List[str] """Compute all the weak and strong coupling variables. Returns: The names of the coupling variables. """ inputs = [] outputs = [] for discipline in self.disciplines: inputs += discipline.get_input_data_names() outputs += discipline.get_output_data_names() return sorted(list(set(inputs) & set(outputs)))
[docs] def output_couplings( self, discipline, # type: MDODiscipline strong=True, # type: bool ): # type: (...) -> List[str] """Compute the output coupling variables of a discipline, either strong or weak. Args: discipline: The discipline. strong: If True, consider the strong couplings. Otherwise, the weak ones. Returns: The names of the output coupling variables. """ output_names = discipline.get_output_data_names() if strong: couplings = self.strong_couplings() else: couplings = self.get_all_couplings() return sorted([name for name in output_names if name in couplings])
[docs] def find_discipline( self, output, # type: str ): # type: (...) -> MDODiscipline """Find which discipline produces a given output. Args: output: The name of an output. Returns: The discipline producing this output, if it exists. Raises: TypeError: If the name of the output is not a string. ValueError: If the output is not an output of the discipline. """ if not isinstance(output, string_types): raise TypeError("Output shall be a string") for discipline in self.disciplines: if discipline.is_output_existing(output): return discipline raise ValueError("{} is not the output of a discipline".format(output))
[docs] def plot_n2_chart( self, file_path="n2.pdf", # type: str show_data_names=True, # type:bool save=True, # type:bool show=False, # type: bool figsize=(15, 10), # type: Tuple[int] open_browser=False, # type:bool ): # type: (...) -> None """Generate a N2 plot for the disciplines. Args: file_path: The name of the file path of the figure. show_data_names: If ``True``, show the names of the coupling data ; otherwise, circles are drawn, which size depend on the number of coupling names. save: If True, save the figure to file_path. show: If True, show the plot. figsize: The width and height of the figure. open_browser: If True, open a browser and display an interactive N2 chart. Raises: ValueError: If there is only 1 discipline. """ output_directory_path = Path(file_path).parent html_file_name = "n2.html" html_file_path = output_directory_path / html_file_name N2HTML(html_file_path, open_browser).from_graph(self.graph) fig = plt.figure(figsize=figsize) plt.grid(True) axe = gca() axe.grid(True, linestyle="-", color="black", lw=1) n_disciplines = len(self.disciplines) if n_disciplines == 1: raise ValueError("N2 Diagrams can be generated for at least 2 disciplines.") ax_ticks = list(range(n_disciplines + 1)) axe.xaxis.set_ticks(ax_ticks) axe.yaxis.set_ticks(ax_ticks) axe.xaxis.set_ticklabels([]) axe.yaxis.set_ticklabels([]) axe.set(xlim=(0, ax_ticks[-1]), ylim=(0, ax_ticks[-1])) axe.tick_params(axis="x", direction="in") axe.tick_params(axis="y", direction="in") fig.tight_layout() for discipline_index, discipline in enumerate(self.disciplines): x_1 = discipline_index x_2 = discipline_index + 1 y_1 = n_disciplines - discipline_index y_2 = n_disciplines - discipline_index - 1 plt.fill( [x_1, x_1, x_2, x_2], [y_1, y_2, y_2, y_1], "limegreen", alpha=0.45 ) discipline_name = plt.text( discipline_index + 0.5, n_disciplines - discipline_index - 0.5, discipline.name, verticalalignment="center", horizontalalignment="center", ) self._check_size_text(discipline_name, fig, n_disciplines) couplings = self.graph.get_disciplines_couplings() max_coupling_length = array([len(coupling[2]) for coupling in couplings]).max() for source, destination, variables in couplings: source_position = self.disciplines.index(source) destination_position = self.disciplines.index(destination) if show_data_names: variables_names = plt.text( destination_position + 0.5, n_disciplines - source_position - 0.5, "\n".join(variables), verticalalignment="center", horizontalalignment="center", ) self._check_size_text(variables_names, fig, n_disciplines) else: circle = plt.Circle( (0.5 + destination_position, n_disciplines - 0.5 - source_position), len(variables) / (3.0 * max_coupling_length), color="blue", ) axe.add_artist(circle) if save: plt.savefig(file_path) if show: plt.show()
@staticmethod def _check_size_text( text, # type: Text figure, # type: Figure n_disciplines, # type: int ): # type: (...)-> None """Adapt the size of the figure based on the size of the text to display. Args: text: The text shown in the N2 matrix. figure: The figure of the N2 matrix. n_disciplines: The number of disciplines to be visible. """ renderer = figure.canvas.get_renderer() bbox = text.get_window_extent(renderer=renderer) width = bbox.width height = bbox.height size_max_box = figure.get_size_inches() * figure.dpi / n_disciplines inches = figure.get_size_inches() if width > size_max_box[0]: width_l = round(width * n_disciplines / figure.dpi) + 1 figure.set_size_inches(width_l, inches[1]) if height > size_max_box[1]: length_l = round(height * n_disciplines / figure.dpi) + 1 figure.set_size_inches(inches[0], length_l)