Source code for wntr.network.model

"""
The wntr.network.model module includes methods to build a water network
model.
"""
import logging
from collections import OrderedDict
from typing import List, Union
from warnings import warn

import networkx as nx
import numpy as np
import pandas as pd
import six
import wntr.epanet
import wntr.network.io
from wntr.utils.ordered_set import OrderedSet

from .base import AbstractModel, Link, LinkStatus, Registry
from .controls import Control, Rule
from .elements import (
    Curve,
    Demands,
    FCValve,
    GPValve,
    HeadPump,
    Junction,
    Pattern,
    PBValve,
    Pipe,
    PowerPump,
    PRValve,
    PSValve,
    Pump,
    Reservoir,
    Source,
    Tank,
    TCValve,
    TimeSeries,
    Valve,
)

from .options import Options
from .io import read_inpfile

logger = logging.getLogger(__name__)


[docs] class WaterNetworkModel(AbstractModel): """ Water network model class. Parameters ------------------- inp_file_name: string (optional) Directory and filename of EPANET inp file to load into the WaterNetworkModel object. """
[docs] def __init__(self, inp_file_name=None): # Network name self.name = None self._references: List[Union[str, dict]] = list() """A list of references that document the source of this model.""" self._options = Options() self._node_reg = NodeRegistry(self) self._link_reg = LinkRegistry(self) self._pattern_reg = PatternRegistry(self) self._curve_reg = CurveRegistry(self) self._controls = OrderedDict() self._sources = SourceRegistry(self) self._msx = None self._node_reg._finalize_(self) self._link_reg._finalize_(self) self._pattern_reg._finalize_(self) self._curve_reg._finalize_(self) self._sources._finalize_(self) # NetworkX Graph to store the pipe connectivity and node coordinates self._labels = None self._inpfile = None if inp_file_name: read_inpfile(inp_file_name, append=self) # To be deleted and/or renamed and/or moved # Time parameters self.sim_time = 0.0 self._prev_sim_time = None # the last time at which results were accepted
def _compare(self, other, level=1): """ Parameters ---------- other: WaterNetworkModel Returns ------- bool """ if ( self.num_junctions != other.num_junctions or self.num_reservoirs != other.num_reservoirs or self.num_tanks != other.num_tanks or self.num_pipes != other.num_pipes or self.num_pumps != other.num_pumps or self.num_valves != other.num_valves ): return False for name, node in self.nodes(): if not node._compare(other.get_node(name)): return False for name, link in self.links(): if not link._compare(other.get_link(name)): return False if level > 0: for name, pat in self.patterns(): if pat != other.get_pattern(name): return False for name, curve in self.curves(): if curve != other.get_curve(name): return False for name, source in self.sources(): if source != other.get_source(name): return False if self.options != other.options: return False for name, control in self.controls(): if not control._compare(other.get_control(name)): return False return True @property def _shifted_time(self): """ Return the time in seconds shifted by the simulation start time (e.g. as specified in the inp file). This is, this is the time since 12 AM on the first day. """ return self.sim_time + self.options.time.start_clocktime @property def _prev_shifted_time(self): """ Return the time in seconds of the previous solve shifted by the simulation start time. That is, this is the time from 12 AM on the first day to the time at the previous hydraulic timestep. """ return self._prev_sim_time + self.options.time.start_clocktime @property def _clock_time(self): """ Return the current time of day in seconds from 12 AM """ return self._shifted_time % (24 * 3600) @property def _clock_day(self): """Return the clock-time day of the simulation""" return int(self._shifted_time / 86400) ### # ### Iteratable attributes @property def options(self): """The model's options object Returns ------- Options """ return self._options @property def nodes(self): """The node registry (as property) or a generator for iteration (as function call) Returns ------- NodeRegistry """ return self._node_reg @property def links(self): """The link registry (as property) or a generator for iteration (as function call) Returns ------- LinkRegistry """ return self._link_reg @property def patterns(self): """The pattern registry (as property) or a generator for iteration (as function call) Returns ------- PatternRegistry """ return self._pattern_reg @property def curves(self): """The curve registry (as property) or a generator for iteration (as function call) Returns ------- CurveRegistry """ return self._curve_reg
[docs] def sources(self): """Returns a generator to iterate over all sources Returns ------- A generator in the format (name, object). """ for source_name, source in self._sources.items(): yield source_name, source
[docs] def controls(self): """Returns a generator to iterate over all controls Returns ------- A generator in the format (name, object). """ for control_name, control in self._controls.items(): yield control_name, control
### # ### Element iterators @property def junctions(self): """Iterator over all junctions""" return self._node_reg.junctions @property def tanks(self): """Iterator over all tanks""" return self._node_reg.tanks @property def reservoirs(self): """Iterator over all reservoirs""" return self._node_reg.reservoirs @property def pipes(self): """Iterator over all pipes""" return self._link_reg.pipes @property def pumps(self): """Iterator over all pumps""" return self._link_reg.pumps @property def valves(self): """Iterator over all valves""" return self._link_reg.valves @property def head_pumps(self): """Iterator over all head-based pumps""" return self._link_reg.head_pumps @property def power_pumps(self): """Iterator over all power pumps""" return self._link_reg.power_pumps @property def prvs(self): """Iterator over all pressure reducing valves (PRVs)""" return self._link_reg.prvs @property def psvs(self): """Iterator over all pressure sustaining valves (PSVs)""" return self._link_reg.psvs @property def pbvs(self): """Iterator over all pressure breaker valves (PBVs)""" return self._link_reg.pbvs @property def tcvs(self): """Iterator over all throttle control valves (TCVs)""" return self._link_reg.tcvs @property def fcvs(self): """Iterator over all flow control valves (FCVs)""" return self._link_reg.fcvs @property def gpvs(self): """Iterator over all general purpose valves (GPVs)""" return self._link_reg.gpvs @property def msx(self): """A multispecies water quality model, if defined""" return self._msx @msx.setter def msx(self, model): if model is None: self._msx = None return from wntr.msx.base import QualityModelBase if not isinstance(model, QualityModelBase): raise TypeError('Expected QualityModelBase (or derived), got {}'.format(type(model))) self._msx = model
[docs] def add_msx_model(self, msx_filename=None): """Add an msx model from a MSX input file (.msx extension)""" from wntr.msx.model import MsxModel self._msx = MsxModel(msx_file_name=msx_filename)
[docs] def remove_msx_model(self): """Remove an msx model from the network""" self._msx = None
""" ### # ### Create blank, unregistered objects (for direct assignment) def new_demand_timeseries_list(self): return Demands(self) def new_timeseries(self): return TimeSeries(self, 0.0) def new_pattern(self): return Pattern(None, time_options=self._options.time) """ ### # ### Add elements to the model
[docs] def add_junction( self, name, base_demand=0.0, demand_pattern=None, elevation=0.0, coordinates=None, demand_category=None ): """ Adds a junction to the water network model Parameters ------------------- name : string Name of the junction. base_demand : float Base demand at the junction. demand_pattern : string or Pattern Name of the demand pattern or the Pattern object elevation : float Elevation of the junction. coordinates : tuple of floats X-Y coordinates of the node location. demand_category : string Name of the demand category """ self._node_reg.add_junction(name, base_demand, demand_pattern, elevation, coordinates, demand_category)
[docs] def add_tank( self, name, elevation=0.0, init_level=3.048, min_level=0.0, max_level=6.096, diameter=15.24, min_vol=0.0, vol_curve=None, overflow=False, coordinates=None, ): """ Adds a tank to the water network model Parameters ------------------- name : string Name of the tank. elevation : float Elevation at the tank. init_level : float Initial tank level. min_level : float Minimum tank level. max_level : float Maximum tank level. diameter : float Tank diameter. min_vol : float Minimum tank volume. vol_curve : str, optional Name of a volume curve overflow : bool Overflow indicator (Always False for the WNTRSimulator) coordinates : tuple of floats, optional X-Y coordinates of the node location. """ self._node_reg.add_tank( name, elevation, init_level, min_level, max_level, diameter, min_vol, vol_curve, overflow, coordinates )
[docs] def add_reservoir(self, name, base_head=0.0, head_pattern=None, coordinates=None): """ Adds a reservoir to the water network model Parameters ---------- name : string Name of the reservoir. base_head : float, optional Base head at the reservoir. head_pattern : string, optional Name of the head pattern. coordinates : tuple of floats, optional X-Y coordinates of the node location. """ self._node_reg.add_reservoir(name, base_head, head_pattern, coordinates)
[docs] def add_pipe( self, name, start_node_name, end_node_name, length=304.8, diameter=0.3048, roughness=100, minor_loss=0.0, initial_status="OPEN", check_valve=False, ): """ Adds a pipe to the water network model Parameters ---------- name : string Name of the pipe. start_node_name : string Name of the start node. end_node_name : string Name of the end node. length : float, optional Length of the pipe. diameter : float, optional Diameter of the pipe. roughness : float, optional Pipe roughness coefficient. minor_loss : float, optional Pipe minor loss coefficient. initial_status : string or LinkStatus, optional Pipe initial status. Options are 'OPEN' or 'CLOSED'. check_valve : bool, optional True if the pipe has a check valve. False if the pipe does not have a check valve. """ self._link_reg.add_pipe( name, start_node_name, end_node_name, length, diameter, roughness, minor_loss, initial_status, check_valve )
[docs] def add_pump( self, name, start_node_name, end_node_name, pump_type="POWER", pump_parameter=50.0, speed=1.0, pattern=None, initial_status="OPEN", ): """ Adds a pump to the water network model Parameters ---------- name : string Name of the pump. start_node_name : string Name of the start node. end_node_name : string Name of the end node. pump_type : string, optional Type of information provided for a pump. Options are 'POWER' or 'HEAD'. pump_parameter : float or string For a POWER pump, the pump power. For a HEAD pump, the head curve name. speed: float Relative speed setting (1.0 is normal speed) pattern: string Name of the speed pattern initial_status : string or LinkStatus Pump initial status. Options are 'OPEN' or 'CLOSED'. """ self._link_reg.add_pump( name, start_node_name, end_node_name, pump_type, pump_parameter, speed, pattern, initial_status )
[docs] def add_valve( self, name, start_node_name, end_node_name, diameter=0.3048, valve_type="PRV", minor_loss=0.0, initial_setting=0.0, initial_status="ACTIVE", ): """ Adds a valve to the water network model Parameters ---------- name : string Name of the valve. start_node_name : string Name of the start node. end_node_name : string Name of the end node. diameter : float, optional Diameter of the valve. valve_type : string, optional Type of valve. Options are 'PRV', 'PSV', 'PBV', 'FCV', 'TCV', and 'GPV' minor_loss : float, optional Pipe minor loss coefficient. initial_setting : float or string, optional Valve initial setting. Pressure setting for PRV, PSV, or PBV. Flow setting for FCV. Loss coefficient for TCV. Name of headloss curve for GPV. initial_status: string or LinkStatus Valve initial status. Options are 'OPEN', 'CLOSED', or 'ACTIVE'. """ self._link_reg.add_valve( name, start_node_name, end_node_name, diameter, valve_type, minor_loss, initial_setting, initial_status )
[docs] def add_pattern(self, name, pattern=None): """ Adds a pattern to the water network model The pattern can be either a list of values (list, numpy array, etc.) or a :class:`~wntr.network.elements.Pattern` object. The Pattern class has options to automatically create certain types of patterns, such as a single, on/off pattern (previously created using the start_time and stop_time arguments to this function) -- see the class documentation for examples. .. warning:: Patterns **must** be added to the model prior to adding any model element that uses the pattern, such as junction demands, sources, etc. Patterns are linked by reference, so changes to a pattern affects all elements using that pattern. .. warning:: Patterns **always** use the global water network model options.time values. Patterns **will not** be resampled to match these values, it is assumed that patterns created using Pattern(...) or Pattern.binary_pattern(...) object used the same pattern timestep value as the global value, and they will be treated accordingly. Parameters ---------- name : string Name of the pattern. pattern : list of floats or Pattern A list of floats that make up the pattern, or a :class:`~wntr.network.elements.Pattern` object. Raises ------ ValueError If adding a pattern with `name` that already exists. """ self._pattern_reg.add_pattern(name, pattern)
[docs] def add_curve(self, name, curve_type, xy_tuples_list): """ Adds a curve to the water network model Parameters ---------- name : string Name of the curve. curve_type : string Type of curve. Options are HEAD, EFFICIENCY, VOLUME, HEADLOSS. xy_tuples_list : list of (x, y) tuples List of X-Y coordinate tuples on the curve. """ self._curve_reg.add_curve(name, curve_type, xy_tuples_list)
[docs] def add_source(self, name, node_name, source_type, quality, pattern=None): """ Adds a source to the water network model Parameters ---------- name : string Name of the source node_name: string Injection node. source_type: string Source type, options = CONCEN, MASS, FLOWPACED, or SETPOINT quality: float Source strength in Mass/Time for MASS and Mass/Volume for CONCEN, FLOWPACED, or SETPOINT pattern: string or Pattern object Pattern name or object """ if pattern and isinstance(pattern, six.string_types): pattern = self.get_pattern(pattern) source = Source(self, name, node_name, source_type, quality, pattern) self._sources[source.name] = source self._pattern_reg.add_usage(source.strength_timeseries.pattern_name, (source.name, "Source")) self._node_reg.add_usage(source.node_name, (source.name, "Source"))
[docs] def add_control(self, name, control_object): """ Adds a control or rule to the water network model Parameters ---------- name : string control object name. control_object : Control or Rule Control or Rule object. """ if name in self._controls: raise ValueError( "The name provided for the control is already used. Please either remove the control with that name first or use a different name for this control." ) self._controls[name] = control_object
### # ### Remove elements from the model
[docs] def remove_node(self, name, with_control=False, force=False): """Removes a node from the water network model""" node = self.get_node(name) if not force: if with_control: x = [] for control_name, control in self._controls.items(): if node in control.requires(): logger.warning( control._control_type_str() + " " + control_name + " is being removed along with node " + name ) x.append(control_name) for i in x: self.remove_control(i) else: for control_name, control in self._controls.items(): if node in control.requires(): raise RuntimeError( "Cannot remove node {0} without first removing control/rule {1}".format(name, control_name) ) self._node_reg.__delitem__(name)
[docs] def remove_pattern(self, name): """Removes a pattern from the water network model""" self._pattern_reg.__delitem__(name)
[docs] def remove_curve(self, name): """Removes a curve from the water network model""" self._curve_reg.__delitem__(name)
[docs] def remove_source(self, name): """Removes a source from the water network model Parameters ---------- name : string The name of the source object to be removed """ logger.warning( "You are deleting a source. This could have unintended \ side effects. If you are replacing values, use get_source(name) \ and modify it instead." ) source = self._sources[name] self._pattern_reg.remove_usage(source.strength_timeseries.pattern_name, (source.name, "Source")) self._node_reg.remove_usage(source.node_name, (source.name, "Source")) del self._sources[name]
[docs] def remove_control(self, name): """Removes a control from the water network model""" del self._controls[name]
def _discard_control(self, name): """Removes a control from the water network model If the control is not present, an exception is not raised. Parameters ---------- name : string The name of the control object to be removed. """ try: del self._controls[name] except KeyError: pass ### # ### Get elements from the model
[docs] def get_node(self, name): """Get a specific node Parameters ---------- name : str The node name Returns ------- Junction, Tank, or Reservoir """ return self._node_reg[name]
[docs] def get_pattern(self, name): """Get a specific pattern Parameters ---------- name : str The pattern name Returns ------- Pattern """ return self._pattern_reg[name]
[docs] def get_curve(self, name): """Get a specific curve Parameters ---------- name : str The curve name Returns ------- Curve """ return self._curve_reg[name]
[docs] def get_source(self, name): """Get a specific source Parameters ---------- name : str The source name Returns ------- Source """ return self._sources[name]
[docs] def get_control(self, name): """Get a specific control or rule Parameters ---------- name : str The control name Returns ------- ctrl: Control or Rule """ return self._controls[name]
### # ### Name lists @property def node_name_list(self): """Get a list of node names Returns ------- list of strings """ return list(self._node_reg.keys()) @property def junction_name_list(self): """Get a list of junction names Returns ------- list of strings """ return list(self._node_reg.junction_names) @property def tank_name_list(self): """Get a list of tanks names Returns ------- list of strings """ return list(self._node_reg.tank_names) @property def reservoir_name_list(self): """Get a list of reservoir names Returns ------- list of strings """ return list(self._node_reg.reservoir_names) @property def link_name_list(self): """Get a list of link names Returns ------- list of strings """ return list(self._link_reg.keys()) @property def pipe_name_list(self): """Get a list of pipe names Returns ------- list of strings """ return list(self._link_reg.pipe_names) @property def pump_name_list(self): """Get a list of pump names (both types included) Returns ------- list of strings """ return list(self._link_reg.pump_names) @property def head_pump_name_list(self): """Get a list of head pump names Returns ------- list of strings """ return list(self._link_reg.head_pump_names) @property def power_pump_name_list(self): """Get a list of power pump names Returns ------- list of strings """ return list(self._link_reg.power_pump_names) @property def valve_name_list(self): """Get a list of valve names (all types included) Returns ------- list of strings """ return list(self._link_reg.valve_names) @property def prv_name_list(self): """Get a list of prv names Returns ------- list of strings """ return list(self._link_reg.prv_names) @property def psv_name_list(self): """Get a list of psv names Returns ------- list of strings """ return list(self._link_reg.psv_names) @property def pbv_name_list(self): """Get a list of pbv names Returns ------- list of strings """ return list(self._link_reg.pbv_names) @property def tcv_name_list(self): """Get a list of tcv names Returns ------- list of strings """ return list(self._link_reg.tcv_names) @property def fcv_name_list(self): """Get a list of fcv names Returns ------- list of strings """ return list(self._link_reg.fcv_names) @property def gpv_name_list(self): """Get a list of gpv names Returns ------- list of strings """ return list(self._link_reg.gpv_names) @property def pattern_name_list(self): """Get a list of pattern names Returns ------- list of strings """ return list(self._pattern_reg.keys()) @property def curve_name_list(self): """Get a list of curve names Returns ------- list of strings """ return list(self._curve_reg.keys()) @property def source_name_list(self): """Get a list of source names Returns ------- list of strings """ return list(self._sources.keys()) @property def control_name_list(self): """Get a list of control/rule names Returns ------- list of strings """ return list(self._controls.keys()) ### # ### Counts @property def num_nodes(self): """The number of nodes""" return len(self._node_reg) @property def num_junctions(self): """The number of junctions""" return len(self._node_reg.junction_names) @property def num_tanks(self): """The number of tanks""" return len(self._node_reg.tank_names) @property def num_reservoirs(self): """The number of reservoirs""" return len(self._node_reg.reservoir_names) @property def num_links(self): """The number of links""" return len(self._link_reg) @property def num_pipes(self): """The number of pipes""" return len(self._link_reg.pipe_names) @property def num_pumps(self): """The number of pumps""" return len(self._link_reg.pump_names) @property def num_valves(self): """The number of valves""" return len(self._link_reg.valve_names) @property def num_patterns(self): """The number of patterns""" return len(self._pattern_reg) @property def num_curves(self): """The number of curves""" return len(self._curve_reg) @property def num_sources(self): """The number of sources""" return len(self._sources) @property def num_controls(self): """The number of controls""" return len(self._controls) ### # ### Helper functions
[docs] def describe(self, level=0): """ Describe number of components in the network model Parameters ---------- level : int (0, 1, or 2) * Level 0 returns the number of Nodes, Links, Patterns, Curves, Sources, and Controls. * Level 1 includes information from Level 0 but divides Nodes into Junctions, Tanks, and Reservoirs, divides Links into Pipes, Pumps, and Valves, and divides Curves into Pump, Efficiency, Headloss, and Volume. * Level 2 includes information from Level 1 but divides Pumps into Head and Power, and divides Valves into PRV, PSV, PBV, TCV, FCV, and GPV. Returns ------- A dictionary with component counts """ d = { "Nodes": self.num_nodes, "Links": self.num_links, "Patterns": self.num_patterns, "Curves": self.num_curves, "Sources": self.num_sources, "Controls": self.num_controls, } if level >= 1: d["Nodes"] = {"Junctions": self.num_junctions, "Tanks": self.num_tanks, "Reservoirs": self.num_reservoirs} d["Links"] = {"Pipes": self.num_pipes, "Pumps": self.num_pumps, "Valves": self.num_valves} d["Curves"] = { "Pump": len(self._curve_reg._pump_curves), "Efficiency": len(self._curve_reg._efficiency_curves), "Headloss": len(self._curve_reg._headloss_curves), "Volume": len(self._curve_reg._volume_curves), } if level >= 2: d["Links"]["Pumps"] = {"Head": len(list(self.head_pumps())), "Power": len(list(self.power_pumps()))} d["Links"]["Valves"] = { "PRV": len(list(self.prvs())), "PSV": len(list(self.psvs())), "PBV": len(list(self.pbvs())), "TCV": len(list(self.tcvs())), "FCV": len(list(self.fcvs())), "GPV": len(list(self.gpvs())), } return d
[docs] def to_dict(self): """ Dictionary representation of the WaterNetworkModel. Returns ------- dict """ return wntr.network.io.to_dict(self)
[docs] def from_dict(self, d: dict): """ Append the model with elements from a dictionary. Parameters ---------- d : dict Dictionary representation of the WaterNetworkModel """ wntr.network.io.from_dict(d, append=self)
[docs] def to_gis(self, crs=None, pumps_as_points=False, valves_as_points=False): """ Convert a WaterNetworkModel into GeoDataFrames Parameters ---------- crs : str, optional Coordinate reference system, by default None pumps_as_points : bool, optional Represent pumps as points (True) or lines (False), by default False valves_as_points : bool, optional Represent valves as points (True) or lines (False), by default False Returns ------- WaterNetworkGIS object that contains junctions, tanks, reservoirs, pipes, pumps, and valves GeoDataFrames """ return wntr.network.io.to_gis(self, crs, pumps_as_points, valves_as_points)
[docs] def from_gis(self, gis_data): """ Append the model with elements from GeoDataFrames Parameters ---------- gis_data : WaterNetworkGIS or dictionary of GeoDataFrames GeoDataFrames containing water network attributes. If gis_data is a dictionary, then the keys are junctions, tanks, reservoirs, pipes, pumps, and valves. If the pumps or valves are Points, they will be converted to Lines with the same start and end node location. Returns ------- WaterNetworkModel """ return wntr.network.io.from_gis(gis_data, append=self)
[docs] def to_graph(self, node_weight=None, link_weight=None, modify_direction=False): """ Convert a WaterNetworkModel into a networkx MultiDiGraph Parameters ---------- node_weight : dict or pandas Series (optional) Node weights link_weight : dict or pandas Series (optional) Link weights. modify_direction : bool (optional) If True, than if the link weight is negative, the link start and end node are switched and the abs(weight) is assigned to the link (this is useful when weighting graphs by flowrate). If False, link direction and weight are not changed. Returns -------- networkx MultiDiGraph """ return wntr.network.io.to_graph(self, node_weight, link_weight, modify_direction)
[docs] def get_graph(self, node_weight=None, link_weight=None, modify_direction=False): """ Convert a WaterNetworkModel into a networkx MultiDiGraph .. deprecated:: 0.5.0 Use ``to_graph()`` instead Parameters ---------- node_weight : dict or pandas Series (optional) Node weights link_weight : dict or pandas Series (optional) Link weights. modify_direction : bool (optional) If True, than if the link weight is negative, the link start and end node are switched and the abs(weight) is assigned to the link (this is useful when weighting graphs by flowrate). If False, link direction and weight are not changed. Returns -------- networkx MultiDiGraph """ warn("wntr.network.WaterNetworkModel.get_graph is deprecated, use wntr.network.WaterNetworkModel.to_graph instead", DeprecationWarning, stacklevel=2) return wntr.network.io.to_graph(self, node_weight, link_weight, modify_direction)
[docs] def assign_demand(self, demand, pattern_prefix="ResetDemand"): """ Assign demands using values in a DataFrame. New demands are specified in a pandas DataFrame indexed by time (in seconds). The method resets junction demands by creating a new demand pattern and using a base demand of 1. The demand pattern is resampled to match the water network model pattern timestep. This method can be used to reset demands in a water network model to demands from a pressure dependent demand simulation. Parameters ---------- demand : pandas DataFrame A pandas DataFrame containing demands (index = time, columns = junction names) pattern_prefix: string Pattern name prefix, default = 'ResetDemand'. The junction name is appended to the prefix to create a new pattern name. If the pattern name already exists, an error is thrown and the user should use a different pattern prefix name. """ for junc_name in demand.columns: # Extract the node demand pattern and resample to match the pattern timestep demand_pattern = demand.loc[:, junc_name] demand_pattern.index = pd.to_timedelta(demand_pattern.index, "s") resample_offset = str(int(self.options.time.pattern_timestep)) + "s" demand_pattern = demand_pattern.resample(resample_offset).mean() / self.options.hydraulic.demand_multiplier # Add the pattern # If the pattern name already exists, this fails pattern_name = pattern_prefix + junc_name self.add_pattern(pattern_name, demand_pattern.tolist()) # Reset base demand junction = self.get_node(junc_name) junction.demand_timeseries_list.clear() junction.demand_timeseries_list.append((1.0, pattern_name))
[docs] def query_node_attribute(self, attribute, operation=None, value=None, node_type=None): """ Query node attributes, for example get all nodes with elevation <= threshold Parameters ---------- attribute: string Node attribute. operation: numpy operator Numpy operator, options include np.greater, np.greater_equal, np.less, np.less_equal, np.equal, np.not_equal. value: float or int Threshold node_type: Node type Node type, options include wntr.network.model.Node, wntr.network.model.Junction, wntr.network.model.Reservoir, wntr.network.model.Tank, or None. Default = None. Note None and wntr.network.model.Node produce the same results. Returns ------- A pandas Series that contains the attribute that satisfies the operation threshold for a given node_type. Notes ----- If operation and value are both None, the Series will contain the attributes for all nodes with the specified attribute. """ node_attribute_dict = {} for name, node in self.nodes(node_type): try: if operation == None and value == None: node_attribute_dict[name] = getattr(node, attribute) else: node_attribute = getattr(node, attribute) if operation(node_attribute, value): node_attribute_dict[name] = node_attribute except AttributeError: pass return pd.Series(node_attribute_dict)
[docs] def convert_controls_to_rules(self, priority=3): """ Convert all controls to rules. Note that for an exact match between controls and rules, the rule timestep must be very small. Parameters ---------- priority : int, optional Rule priority, default is 3. """ for name in self.control_name_list: control = self.get_control(name) if isinstance(control, Control): act = control.actions()[0] cond = control.condition rule = Rule(cond, act, priority=priority) self.add_control(name.replace(" ", "_") + "_Rule", rule) self.remove_control(name)
[docs] def reset_initial_values(self): """ Resets all initial values in the network """ #### TODO: move reset conditions to /sim self.sim_time = 0.0 self._prev_sim_time = None for name, node in self.nodes(Junction): node._head = None node._demand = None node._pressure = None node._leak_demand = None node._leak_status = False node._is_isolated = False for name, node in self.nodes(Tank): node._head = node.init_level + node.elevation node._prev_head = node.head node._demand = None node._leak_demand = None node._leak_status = False node._is_isolated = False for name, node in self.nodes(Reservoir): node._head = None # node.head_timeseries.base_value node._demand = None node._leak_demand = None node._is_isolated = False for name, link in self.links(Pipe): link._user_status = link.initial_status link._setting = link.initial_setting link._internal_status = LinkStatus.Active link._is_isolated = False link._flow = None link._prev_setting = None for name, link in self.links(Pump): link._user_status = link.initial_status link._setting = link.initial_setting link._internal_status = LinkStatus.Active link._is_isolated = False link._flow = None if isinstance(link, PowerPump): link.power = link._base_power link._prev_setting = None for name, link in self.links(Valve): link._user_status = link.initial_status link._setting = link.initial_setting link._internal_status = LinkStatus.Active link._is_isolated = False link._flow = None link._prev_setting = None for name, control in self.controls(): control._reset()
[docs] class PatternRegistry(Registry): """A registry for patterns.""" def _finalize_(self, model): super()._finalize_(model) self._pattern_reg = None
[docs] class DefaultPattern(object): """An object that always points to the current default pattern for a model"""
[docs] def __init__(self, options): self._options = options
def __str__(self): return str(self._options.hydraulic.pattern) if self._options.hydraulic.pattern is not None else "" def __repr__(self): return "DefaultPattern()" @property def name(self): """The name of the default pattern, or ``None`` if no pattern is assigned""" return str(self._options.hydraulic.pattern) if self._options.hydraulic.pattern is not None else ""
def __getitem__(self, key): try: return super(PatternRegistry, self).__getitem__(key) except KeyError: return None
[docs] def add_pattern(self, name, pattern=None): """ Adds a pattern to the water network model. The pattern can be either a list of values (list, numpy array, etc.) or a :class:`~wntr.network.elements.Pattern` object. The Pattern class has options to automatically create certain types of patterns, such as a single, on/off pattern .. warning:: Patterns **must** be added to the model prior to adding any model element that uses the pattern, such as junction demands, sources, etc. Patterns are linked by reference, so changes to a pattern affects all elements using that pattern. .. warning:: Patterns **always** use the global water network model options.time values. Patterns **will not** be resampled to match these values, it is assumed that patterns created using Pattern(...) or Pattern.binary_pattern(...) object used the same pattern timestep value as the global value, and they will be treated accordingly. Parameters ---------- name : string Name of the pattern. pattern : list of floats or Pattern A list of floats that make up the pattern, or a :class:`~wntr.network.elements.Pattern` object. Raises ------ ValueError If adding a pattern with `name` that already exists. """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert isinstance(pattern, (list, np.ndarray, Pattern)), "pattern must be a list or Pattern" if not isinstance(pattern, Pattern): pattern = Pattern(name, multipliers=pattern, time_options=self._options.time) else: # elif pattern.time_options is None: pattern.time_options = self._options.time if pattern.name in self._data.keys(): raise ValueError("Pattern name already exists") self[name] = pattern
@property def default_pattern(self): """A new default pattern object""" return self.DefaultPattern(self._options)
[docs] class CurveRegistry(Registry): """A registry for curves."""
[docs] def __init__(self, model): super(CurveRegistry, self).__init__(model) self._pump_curves = OrderedSet() self._efficiency_curves = OrderedSet() self._headloss_curves = OrderedSet() self._volume_curves = OrderedSet()
def _finalize_(self, model): super()._finalize_(model) self._curve_reg = None def __setitem__(self, key, value): if not isinstance(key, six.string_types): raise ValueError("Registry keys must be strings") self._data[key] = value if value is not None: self.set_curve_type(key, value.curve_type)
[docs] def set_curve_type(self, key, curve_type): """ Sets curve type. WARNING -- this does not check to make sure key is typed before assigning it - you could end up with a curve that is used for more than one type""" if curve_type is None: return curve_type = curve_type.upper() if curve_type == "HEAD": self._pump_curves.add(key) elif curve_type == "HEADLOSS": self._headloss_curves.add(key) elif curve_type == "VOLUME": self._volume_curves.add(key) elif curve_type == "EFFICIENCY": self._efficiency_curves.add(key) else: raise ValueError("curve_type must be HEAD, HEADLOSS, VOLUME, or EFFICIENCY")
[docs] def add_curve(self, name, curve_type, xy_tuples_list): """ Adds a curve to the water network model. Parameters ---------- name : string Name of the curve. curve_type : string Type of curve. Options are HEAD, EFFICIENCY, VOLUME, HEADLOSS. xy_tuples_list : list of (x, y) tuples List of X-Y coordinate tuples on the curve. """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert isinstance(curve_type, (type(None), str)), "curve_type must be a string" assert isinstance(xy_tuples_list, (list, np.ndarray)), "xy_tuples_list must be a list of (x,y) tuples" curve = Curve(name, curve_type, xy_tuples_list) self[name] = curve
[docs] def untyped_curves(self): """Generator to get all curves without type Yields ------ name : str The name of the curve curve : Curve The untyped curve object """ defined = set(self._data.keys()) untyped = defined.difference( self._pump_curves, self._efficiency_curves, self._headloss_curves, self._volume_curves ) for key in untyped: yield key, self._data[key]
@property def untyped_curve_names(self): """List of names of all curves without types""" defined = set(self._data.keys()) untyped = defined.difference( self._pump_curves, self._efficiency_curves, self._headloss_curves, self._volume_curves ) return list(untyped)
[docs] def pump_curves(self): """Generator to get all pump curves Yields ------ name : str The name of the curve curve : Curve The pump curve object """ for key in self._pump_curves: yield key, self._data[key]
@property def pump_curve_names(self): """List of names of all pump curves""" return list(self._pump_curves)
[docs] def efficiency_curves(self): """Generator to get all efficiency curves Yields ------ name : str The name of the curve curve : Curve The efficiency curve object """ for key in self._efficiency_curves: yield key, self._data[key]
@property def efficiency_curve_names(self): """List of names of all efficiency curves""" return list(self._efficiency_curves)
[docs] def headloss_curves(self): """Generator to get all headloss curves Yields ------ name : str The name of the curve curve : Curve The headloss curve object """ for key in self._headloss_curves: yield key, self._data[key]
@property def headloss_curve_names(self): """List of names of all headloss curves""" return list(self._headloss_curves)
[docs] def volume_curves(self): """Generator to get all volume curves Yields ------ name : str The name of the curve curve : Curve The volume curve object """ for key in self._volume_curves: yield key, self._data[key]
@property def volume_curve_names(self): """List of names of all volume curves""" return list(self._volume_curves)
[docs] class SourceRegistry(Registry): """A registry for sources.""" def _finalize_(self, model): super()._finalize_(model) self._sources = None def __delitem__(self, key): try: if self._usage and key in self._usage and len(self._usage[key]) > 0: raise RuntimeError( "cannot remove %s %s, still used by %s" % (self.__class__.__name__, key, self._usage[key]) ) elif key in self._usage: self._usage.pop(key) source = self._data.pop(key) self._pattern_reg.remove_usage(source.strength_timeseries.pattern_name, (source.name, "Source")) self._node_reg.remove_usage(source.node_name, (source.name, "Source")) return source except KeyError: # Do not raise an exception if there is no key of that name return
[docs] class NodeRegistry(Registry): """A registry for nodes."""
[docs] def __init__(self, model): super(NodeRegistry, self).__init__(model) self._junctions = OrderedSet() self._reservoirs = OrderedSet() self._tanks = OrderedSet()
def _finalize_(self, model): super()._finalize_(model) self._node_reg = None def __setitem__(self, key, value): if not isinstance(key, six.string_types): raise ValueError("Registry keys must be strings") self._data[key] = value if isinstance(value, Junction): self._junctions.add(key) elif isinstance(value, Tank): self._tanks.add(key) elif isinstance(value, Reservoir): self._reservoirs.add(key) def __delitem__(self, key): try: if self._usage and key in self._usage and len(self._usage[key]) > 0: raise RuntimeError( "cannot remove %s %s, still used by %s" % (self.__class__.__name__, key, str(self._usage[key])) ) elif key in self._usage: self._usage.pop(key) node = self._data.pop(key) self._junctions.discard(key) self._reservoirs.discard(key) self._tanks.discard(key) if isinstance(node, Junction): for pat_name in node.demand_timeseries_list.pattern_list(): if pat_name: self._curve_reg.remove_usage(pat_name, (node.name, "Junction")) if isinstance(node, Reservoir) and node.head_pattern_name: self._curve_reg.remove_usage(node.head_pattern_name, (node.name, "Reservoir")) if isinstance(node, Tank) and node.vol_curve_name: self._curve_reg.remove_usage(node.vol_curve_name, (node.name, "Tank")) return node except KeyError: return def __call__(self, node_type=None): """ Returns a generator to iterate over all nodes of a specific node type. If no node type is specified, the generator iterates over all nodes. Parameters ---------- node_type: Node type Node type, options include wntr.network.model.Node, wntr.network.model.Junction, wntr.network.model.Reservoir, wntr.network.model.Tank, or None. Default = None. Note None and wntr.network.model.Node produce the same results. Returns ------- A generator in the format (name, object). """ if node_type == None: for node_name, node in self._data.items(): yield node_name, node elif node_type == Junction: for node_name in self._junctions: yield node_name, self._data[node_name] elif node_type == Tank: for node_name in self._tanks: yield node_name, self._data[node_name] elif node_type == Reservoir: for node_name in self._reservoirs: yield node_name, self._data[node_name] else: raise RuntimeError("node_type, " + str(node_type) + ", not recognized.")
[docs] def add_junction( self, name, base_demand=0.0, demand_pattern=None, elevation=0.0, coordinates=None, demand_category=None, emitter_coeff=None, initial_quality=None, ): """ Adds a junction to the water network model. Parameters ------------------- name : string Name of the junction. base_demand : float Base demand at the junction. demand_pattern : string or Pattern Name of the demand pattern or the Pattern object elevation : float Elevation of the junction. coordinates : tuple of floats, optional X-Y coordinates of the node location. demand_category : str, optional Category to the **base** demand emitter_coeff : float, optional Emitter coefficient initial_quality : float, optional Initial quality at this junction """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert isinstance(base_demand, (int, float)), "base_demand must be a float" assert isinstance( demand_pattern, (type(None), str, PatternRegistry.DefaultPattern, Pattern) ), "demand_pattern must be a string or Pattern" assert isinstance(elevation, (int, float)), "elevation must be a float" assert isinstance(coordinates, (type(None), (tuple, list,))), "coordinates must be a tuple" assert isinstance(demand_category, (type(None), str)), "demand_category must be a string" assert isinstance(emitter_coeff, (type(None), int, float)), "emitter_coeff must be a float" assert isinstance(initial_quality, (type(None), int, float)), "initial_quality must be a float" base_demand = float(base_demand) elevation = float(elevation) junction = Junction(name, self) junction.elevation = elevation junction.add_demand(base_demand, demand_pattern, demand_category) self[name] = junction if coordinates is not None: junction.coordinates = coordinates if emitter_coeff is not None: junction.emitter_coefficient = emitter_coeff if initial_quality is not None: junction.initial_quality = initial_quality
[docs] def add_tank( self, name, elevation=0.0, init_level=3.048, min_level=0.0, max_level=6.096, diameter=15.24, min_vol=0.0, vol_curve=None, overflow=False, coordinates=None, ): """ Adds a tank to the water network model. Parameters ------------------- name : string Name of the tank. elevation : float Elevation at the tank. init_level : float Initial tank level. min_level : float Minimum tank level. max_level : float Maximum tank level. diameter : float Tank diameter of a cylindrical tank (only used when the volume curve is None) min_vol : float Minimum tank volume (only used when the volume curve is None) vol_curve : string, optional Name of a volume curve. The volume curve overrides the tank diameter and minimum volume. overflow : bool, optional Overflow indicator (Always False for the WNTRSimulator) coordinates : tuple of floats, optional X-Y coordinates of the node location. """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert isinstance(elevation, (int, float)), "elevation must be a float" assert isinstance(init_level, (int, float)), "init_level must be a float" assert isinstance(min_level, (int, float)), "min_level must be a float" assert isinstance(max_level, (int, float)), "max_level must be a float" assert isinstance(diameter, (int, float)), "diameter must be a float" assert isinstance(min_vol, (int, float)), "min_vol must be a float" assert isinstance(vol_curve, (type(None), str)), "vol_curve must be a string" assert isinstance(overflow, (type(None), str, bool, int)), "overflow must be a bool, 'YES' or 'NO, or 0 or 1" assert isinstance(coordinates, (type(None), (tuple,list,))), "coordinates must be a tuple" elevation = float(elevation) init_level = float(init_level) min_level = float(min_level) max_level = float(max_level) diameter = float(diameter) min_vol = float(min_vol) if init_level < min_level: raise ValueError("Initial tank level must be greater than or equal to the tank minimum level.") if init_level > max_level: raise ValueError("Initial tank level must be less than or equal to the tank maximum level.") if vol_curve is not None and vol_curve != "*": if not isinstance(vol_curve, six.string_types): raise ValueError("Volume curve name must be a string") elif not vol_curve in self._curve_reg.volume_curve_names: raise ValueError( "The volume curve " + vol_curve + " is not one of the curves in the " + "list of volume curves. Valid volume curves are:" + str(self._curve_reg.volume_curve_names) ) vcurve = np.array(self._curve_reg[vol_curve].points) if min_level < vcurve[0, 0]: raise ValueError( ( "The volume curve " + vol_curve + " has a minimum value ({0:5.2f}) \n" + 'greater than the minimum level for tank "' + name + '" ({1:5.2f})\n' + "please correct the user input." ).format(vcurve[0, 0], min_level) ) elif max_level > vcurve[-1, 0]: raise ValueError( ( "The volume curve " + vol_curve + " has a maximum value ({0:5.2f}) \n" + 'less than the maximum level for tank "' + name + '" ({1:5.2f})\n' + "please correct the user input." ).format(vcurve[-1, 0], max_level) ) tank = Tank(name, self) tank.elevation = elevation tank.init_level = init_level tank.min_level = min_level tank.max_level = max_level tank.diameter = diameter tank.min_vol = min_vol tank.vol_curve_name = vol_curve tank.overflow = overflow self[name] = tank if coordinates is not None: tank.coordinates = coordinates
[docs] def add_reservoir(self, name, base_head=0.0, head_pattern=None, coordinates=None): """ Adds a reservoir to the water network model. Parameters ---------- name : string Name of the reservoir. base_head : float, optional Base head at the reservoir. head_pattern : string, optional Name of the head pattern. coordinates : tuple of floats, optional X-Y coordinates of the node location. """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert isinstance(base_head, (int, float)), "base_head must be float" assert isinstance(head_pattern, (type(None), str)), "head_pattern must be a string" assert isinstance(coordinates, (type(None), (tuple, list))), "coordinates must be a tuple" base_head = float(base_head) reservoir = Reservoir(name, self) reservoir.base_head = base_head reservoir.head_pattern_name = head_pattern self[name] = reservoir if coordinates is not None: reservoir.coordinates = coordinates
@property def junction_names(self): """List of names of all junctions""" return self._junctions @property def tank_names(self): """List of names of all junctions""" return self._tanks @property def reservoir_names(self): """List of names of all junctions""" return self._reservoirs
[docs] def junctions(self): """Generator to get all junctions Yields ------ name : str The name of the junction node : Junction The junction object """ for node_name in self._junctions: yield node_name, self._data[node_name]
[docs] def tanks(self): """Generator to get all tanks Yields ------ name : str The name of the tank node : Tank The tank object """ for node_name in self._tanks: yield node_name, self._data[node_name]
[docs] def reservoirs(self): """Generator to get all reservoirs Yields ------ name : str The name of the reservoir node : Reservoir The reservoir object """ for node_name in self._reservoirs: yield node_name, self._data[node_name]
[docs] class LinkRegistry(Registry): """A registry for links.""" __subsets = [ "_pipes", "_pumps", "_head_pumps", "_power_pumps", "_prvs", "_psvs", "_pbvs", "_tcvs", "_fcvs", "_gpvs", "_valves", ]
[docs] def __init__(self, model): super(LinkRegistry, self).__init__(model) self._pipes = OrderedSet() self._pumps = OrderedSet() self._head_pumps = OrderedSet() self._power_pumps = OrderedSet() self._prvs = OrderedSet() self._psvs = OrderedSet() self._pbvs = OrderedSet() self._tcvs = OrderedSet() self._fcvs = OrderedSet() self._gpvs = OrderedSet() self._valves = OrderedSet()
def _finalize_(self, model): super()._finalize_(model) self._link_reg = None def __setitem__(self, key, value): if not isinstance(key, six.string_types): raise ValueError("Registry keys must be strings") self._data[key] = value if isinstance(value, Pipe): self._pipes.add(key) elif isinstance(value, Pump): self._pumps.add(key) if isinstance(value, HeadPump): self._head_pumps.add(key) elif isinstance(value, PowerPump): self._power_pumps.add(key) elif isinstance(value, Valve): self._valves.add(key) if isinstance(value, PRValve): self._prvs.add(key) elif isinstance(value, PSValve): self._psvs.add(key) elif isinstance(value, PBValve): self._pbvs.add(key) elif isinstance(value, TCValve): self._tcvs.add(key) elif isinstance(value, FCValve): self._fcvs.add(key) elif isinstance(value, GPValve): self._gpvs.add(key) def __delitem__(self, key): try: if self._usage and key in self._usage and len(self._usage[key]) > 0: raise RuntimeError( "cannot remove %s %s, still used by %s", self.__class__.__name__, key, self._usage[key] ) elif key in self._usage: self._usage.pop(key) link = self._data.pop(key) self._node_reg.remove_usage(link.start_node_name, (link.name, link.link_type)) self._node_reg.remove_usage(link.end_node_name, (link.name, link.link_type)) if isinstance(link, GPValve): self._curve_reg.remove_usage(link.headloss_curve_name, (link.name, "Valve")) if isinstance(link, Pump): self._curve_reg.remove_usage(link.speed_pattern_name, (link.name, "Pump")) if isinstance(link, HeadPump): self._curve_reg.remove_usage(link.pump_curve_name, (link.name, "Pump")) for ss in self.__subsets: # Go through the _pipes, _prvs, ..., and remove this link getattr(self, ss).discard(key) return link except KeyError: return def __call__(self, link_type=None): """ Returns a generator to iterate over all nodes of a specific node type. If no node type is specified, the generator iterates over all nodes. Parameters ---------- node_type: Node type Node type, options include wntr.network.model.Node, wntr.network.model.Junction, wntr.network.model.Reservoir, wntr.network.model.Tank, or None. Default = None. Note None and wntr.network.model.Node produce the same results. Returns ------- A generator in the format (name, object). """ if link_type == None: for name, node in self._data.items(): yield name, node elif link_type == Pipe: for name in self._pipes: yield name, self._data[name] elif link_type == Pump: for name in self._pumps: yield name, self._data[name] elif link_type == Valve: for name in self._valves: yield name, self._data[name] else: raise RuntimeError("link_type, " + str(link_type) + ", not recognized.")
[docs] def add_pipe( self, name, start_node_name, end_node_name, length=304.8, diameter=0.3048, roughness=100, minor_loss=0.0, initial_status="OPEN", check_valve=False, ): """ Adds a pipe to the water network model. Parameters ---------- name : string Name of the pipe. start_node_name : string Name of the start node. end_node_name : string Name of the end node. length : float, optional Length of the pipe. diameter : float, optional Diameter of the pipe. roughness : float, optional Pipe roughness coefficient. minor_loss : float, optional Pipe minor loss coefficient. initial_status : string, optional Pipe initial status. Options are 'OPEN' or 'CLOSED'. check_valve : bool, optional True if the pipe has a check valve. False if the pipe does not have a check valve. """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert ( isinstance(start_node_name, str) and len(start_node_name) < 32 and start_node_name.find(" ") == -1 ), "start_node_name must be a string with less than 32 characters and contain no spaces" assert ( isinstance(end_node_name, str) and len(end_node_name) < 32 and end_node_name.find(" ") == -1 ), "end_node_name must be a string with less than 32 characters and contain no spaces" assert isinstance(length, (int, float)), "length must be a float" assert isinstance(diameter, (int, float)), "diameter must be a float" assert isinstance(roughness, (int, float)), "roughness must be a float" assert isinstance(minor_loss, (int, float)), "minor_loss must be a float" assert isinstance(initial_status, (int, str, LinkStatus)), "initial_status must be an int, string or LinkStatus" assert isinstance(check_valve, (str, int, bool)), "check_valve must be a Boolean" check_valve = bool(int(check_valve)) length = float(length) diameter = float(diameter) roughness = float(roughness) minor_loss = float(minor_loss) if isinstance(initial_status, str): initial_status = LinkStatus[initial_status] pipe = Pipe(name, start_node_name, end_node_name, self) pipe.length = length pipe.diameter = diameter pipe.roughness = roughness pipe.minor_loss = minor_loss pipe.initial_status = initial_status pipe._user_status = initial_status pipe.check_valve = check_valve self[name] = pipe
[docs] def add_pump( self, name, start_node_name, end_node_name, pump_type="POWER", pump_parameter=50.0, speed=1.0, pattern=None, initial_status="OPEN", ): """ Adds a pump to the water network model. Parameters ---------- name : string Name of the pump. start_node_name : string Name of the start node. end_node_name : string Name of the end node. pump_type : string, optional Type of information provided for a pump. Options are 'POWER' or 'HEAD'. pump_parameter : float or string For a POWER pump, the pump power (float). For a HEAD pump, the head curve name (string). speed: float Relative speed setting (1.0 is normal speed) pattern: string Name of the speed pattern initial_status: str or LinkStatus Pump initial status. Options are 'OPEN' or 'CLOSED'. """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert ( isinstance(start_node_name, str) and len(start_node_name) < 32 and start_node_name.find(" ") == -1 ), "start_node_name must be a string with less than 32 characters and contain no spaces" assert ( isinstance(end_node_name, str) and len(end_node_name) < 32 and end_node_name.find(" ") == -1 ), "end_node_name must be a string with less than 32 characters and contain no spaces" assert isinstance(pump_type, str), "pump_type must be a string" assert isinstance(pump_parameter, (int, float, str)), "pump_parameter must be a float or string" assert isinstance(speed, (int, float)), "speed must be a float" assert isinstance(pattern, (type(None), str)), "pattern must be a string" assert isinstance(initial_status, (int, str, LinkStatus)), "initial_status must be an int, string or LinkStatus" if isinstance(initial_status, str): initial_status = LinkStatus[initial_status] if pump_type.upper() == "POWER": pump = PowerPump(name, start_node_name, end_node_name, self) pump.power = pump_parameter elif pump_type.upper() == "HEAD": pump = HeadPump(name, start_node_name, end_node_name, self) pump.pump_curve_name = pump_parameter else: raise ValueError('pump_type must be "POWER" or "HEAD"') pump.base_speed = speed pump.initial_status = initial_status pump.speed_pattern_name = pattern self[name] = pump
[docs] def add_valve( self, name, start_node_name, end_node_name, diameter=0.3048, valve_type="PRV", minor_loss=0.0, initial_setting=0.0, initial_status="ACTIVE", ): """ Adds a valve to the water network model. Parameters ---------- name : string Name of the valve. start_node_name : string Name of the start node. end_node_name : string Name of the end node. diameter : float, optional Diameter of the valve. valve_type : string, optional Type of valve. Options are 'PRV', 'PSV', 'PBV', 'FCV', 'TCV', and 'GPV' minor_loss : float, optional Pipe minor loss coefficient. initial_setting : float or string, optional Valve initial setting. Pressure setting for PRV, PSV, or PBV. Flow setting for FCV. Loss coefficient for TCV. Name of headloss curve for GPV. initial_status: string or LinkStatus Valve initial status. Options are 'OPEN', 'CLOSED', or 'ACTIVE' """ assert ( isinstance(name, str) and len(name) < 32 and name.find(" ") == -1 ), "name must be a string with less than 32 characters and contain no spaces" assert ( isinstance(start_node_name, str) and len(start_node_name) < 32 and start_node_name.find(" ") == -1 ), "start_node_name must be a string with less than 32 characters and contain no spaces" assert ( isinstance(end_node_name, str) and len(end_node_name) < 32 and end_node_name.find(" ") == -1 ), "end_node_name must be a string with less than 32 characters and contain no spaces" assert isinstance(diameter, (int, float)), "diameter must be a float" assert isinstance(valve_type, str), "valve_type must be a string" assert isinstance(minor_loss, (int, float)), "minor_loss must be a float" assert isinstance(initial_setting, (int, float, str)), "initial_setting must be a float or string" assert isinstance(initial_status, (str, LinkStatus)), "initial_status must be a string or LinkStatus" if isinstance(initial_status, str): initial_status = LinkStatus[initial_status] start_node = self._node_reg[start_node_name] end_node = self._node_reg[end_node_name] valve_type = valve_type.upper() # A PRV, PSV or FCV cannot be directly connected to a reservoir or tank (use a length of pipe to separate the two) if valve_type in ["PRV", "PSV", "FCV"]: if type(start_node) == Tank or type(end_node) == Tank: msg = ( "%ss cannot be directly connected to a tank. Add a pipe to separate the valve from the tank." % valve_type ) logger.error(msg) raise RuntimeError(msg) if type(start_node) == Reservoir or type(end_node) == Reservoir: msg = ( "%ss cannot be directly connected to a reservoir. Add a pipe to separate the valve from the reservoir." % valve_type ) logger.error(msg) raise RuntimeError(msg) # TODO check the following: PRVs cannot share the same downstream node or be linked in series # TODO check the following: Two PSVs cannot share the same upstream node or be linked in series # TODO check the following: A PSV cannot be connected to the downstream node of a PRV if valve_type == "PRV": valve = PRValve(name, start_node_name, end_node_name, self) valve.initial_setting = initial_setting valve._setting = initial_setting elif valve_type == "PSV": valve = PSValve(name, start_node_name, end_node_name, self) valve.initial_setting = initial_setting valve._setting = initial_setting elif valve_type == "PBV": valve = PBValve(name, start_node_name, end_node_name, self) valve.initial_setting = initial_setting valve._setting = initial_setting elif valve_type == "FCV": valve = FCValve(name, start_node_name, end_node_name, self) valve.initial_setting = initial_setting valve._setting = initial_setting elif valve_type == "TCV": valve = TCValve(name, start_node_name, end_node_name, self) valve.initial_setting = initial_setting valve._setting = initial_setting elif valve_type == "GPV": valve = GPValve(name, start_node_name, end_node_name, self) valve.headloss_curve_name = initial_setting valve.initial_status = initial_status valve.diameter = diameter valve.minor_loss = minor_loss self[name] = valve
[docs] def check_valves(self): """Generator to get all pipes with check valves Yields ------ name : str The name of the pipe link : Pipe The pipe object """ for name in self._pipes: if self._data[name].check_valve: yield name
@property def pipe_names(self): """A list of all pipe names""" return self._pipes @property def valve_names(self): """A list of all valve names""" return self._valves @property def pump_names(self): """A list of all pump names""" return self._pumps @property def head_pump_names(self): """A list of all head pump names""" return self._head_pumps @property def power_pump_names(self): """A list of all power pump names""" return self._power_pumps @property def prv_names(self): """A list of all prv names""" return self._prvs @property def psv_names(self): """A list of all psv names""" return self._psvs @property def pbv_names(self): """A list of all pbv names""" return self._pbvs @property def tcv_names(self): """A list of all tcv names""" return self._tcvs @property def fcv_names(self): """A list of all fcv names""" return self._fcvs @property def gpv_names(self): """A list of all gpv names""" return self._gpvs
[docs] def pipes(self): """Generator to get all pipes Yields ------ name : str The name of the pipe link : Pipe The pipe object """ for name in self._pipes: yield name, self._data[name]
[docs] def pumps(self): """Generator to get all pumps Yields ------ name : str The name of the pump link : Pump The pump object """ for name in self._pumps: yield name, self._data[name]
[docs] def valves(self): """Generator to get all valves Yields ------ name : str The name of the valve link : Valve The valve object """ for name in self._valves: yield name, self._data[name]
[docs] def head_pumps(self): """Generator to get all head pumps Yields ------ name : str The name of the pump link : HeadPump The pump object """ for name in self._head_pumps: yield name, self._data[name]
[docs] def power_pumps(self): """Generator to get all power pumps Yields ------ name : str The name of the pump link : PowerPump The pump object """ for name in self._power_pumps: yield name, self._data[name]
[docs] def prvs(self): """Generator to get all PRVs Yields ------ name : str The name of the valve link : PRValve The valve object """ for name in self._prvs: yield name, self._data[name]
[docs] def psvs(self): """Generator to get all PSVs Yields ------ name : str The name of the valve link : PSValve The valve object """ for name in self._psvs: yield name, self._data[name]
[docs] def pbvs(self): """Generator to get all PBVs Yields ------ name : str The name of the valve link : PBValve The valve object """ for name in self._pbvs: yield name, self._data[name]
[docs] def tcvs(self): """Generator to get all TCVs Yields ------ name : str The name of the valve link : TCValve The valve object """ for name in self._tcvs: yield name, self._data[name]
[docs] def fcvs(self): """Generator to get all FCVs Yields ------ name : str The name of the valve link : FCValve The valve object """ for name in self._fcvs: yield name, self._data[name]
[docs] def gpvs(self): """Generator to get all GPVs Yields ------ name : str The name of the valve link : GPValve The valve object """ for name in self._gpvs: yield name, self._data[name]