From 3cfe79bca5f94eeb7a3080b8c6a4bc38d6382f77 Mon Sep 17 00:00:00 2001 From: MarcusHolly Date: Thu, 19 Dec 2024 17:00:12 -0500 Subject: [PATCH 1/2] Add scaler for clarifier --- watertap/unit_models/clarifier.py | 107 +++++- watertap/unit_models/tests/test_clarifier.py | 344 ++++++++++++++++++- 2 files changed, 449 insertions(+), 2 deletions(-) diff --git a/watertap/unit_models/clarifier.py b/watertap/unit_models/clarifier.py index a5f60261f3..1081aa93da 100644 --- a/watertap/unit_models/clarifier.py +++ b/watertap/unit_models/clarifier.py @@ -22,8 +22,10 @@ from idaes.core.util.tables import create_stream_table_dataframe import idaes.logger as idaeslog +from idaes.core.scaling import CustomScalerBase, ConstraintScalingScheme from pyomo.environ import ( + Constraint, Var, Param, units as pyunits, @@ -40,12 +42,115 @@ _log = idaeslog.getLogger(__name__) +class ClarifierScaler(CustomScalerBase): + """ + Default modular scaler for the clarifier unit model. + This Scaler relies on the associated property and reaction packages, + either through user provided options (submodel_scalers argument) or by default + Scalers assigned to the packages. + """ + + DEFAULT_SCALING_FACTORS = { + "surface_area": 1e-3, + } + + def variable_scaling_routine( + self, model, overwrite: bool = False, submodel_scalers: dict = None + ): + """ + Routine to apply scaling factors to variables in model. + Args: + model: model to be scaled + overwrite: whether to overwrite existing scaling factors + submodel_scalers: dict of Scalers to use for sub-models, keyed by submodel local name + Returns: + None + """ + # Call scaling methods for sub-models + self.call_submodel_scaler_method( + submodel=model.mixed_state, + method="variable_scaling_routine", + submodel_scalers=submodel_scalers, + overwrite=overwrite, + ) + self.propagate_state_scaling( + target_state=model.underflow_state, + source_state=model.mixed_state, + overwrite=overwrite, + ) + self.propagate_state_scaling( + target_state=model.effluent_state, + source_state=model.mixed_state, + overwrite=overwrite, + ) + + self.call_submodel_scaler_method( + submodel=model.underflow_state, + method="variable_scaling_routine", + submodel_scalers=submodel_scalers, + overwrite=overwrite, + ) + self.call_submodel_scaler_method( + submodel=model.effluent_state, + method="variable_scaling_routine", + submodel_scalers=submodel_scalers, + overwrite=overwrite, + ) + + # Scale unit level variables + self.scale_variable_by_default(model.surface_area, overwrite=overwrite) + + def constraint_scaling_routine( + self, model, overwrite: bool = False, submodel_scalers: dict = None + ): + """ + Routine to apply scaling factors to constraints in model. + Submodel Scalers are called for the property and reaction blocks. All other constraints + are scaled using the inverse maximum scheme. + Args: + model: model to be scaled + overwrite: whether to overwrite existing scaling factors + submodel_scalers: dict of Scalers to use for sub-models, keyed by submodel local name + Returns: + None + """ + # Call scaling methods for sub-models + self.call_submodel_scaler_method( + submodel=model.mixed_state, + method="constraint_scaling_routine", + submodel_scalers=submodel_scalers, + overwrite=overwrite, + ) + self.call_submodel_scaler_method( + submodel=model.underflow_state, + method="constraint_scaling_routine", + submodel_scalers=submodel_scalers, + overwrite=overwrite, + ) + self.call_submodel_scaler_method( + submodel=model.effluent_state, + method="constraint_scaling_routine", + submodel_scalers=submodel_scalers, + overwrite=overwrite, + ) + + # Scale unit level constraints + for c in model.component_data_objects(Constraint, descend_into=False): + self.scale_constraint_by_nominal_value( + c, + scheme=ConstraintScalingScheme.inverseMaximum, + overwrite=overwrite, + ) + + @declare_process_block_class("Clarifier") class ClarifierData(SeparatorData): """ - Thickener unit model for BSM2 + Clarifier unit model for BSM2 """ + default_scaler = ClarifierScaler + CONFIG = SeparatorData.CONFIG() CONFIG.outlet_list = ["underflow", "overflow"] CONFIG.split_basis = SplittingType.componentFlow diff --git a/watertap/unit_models/tests/test_clarifier.py b/watertap/unit_models/tests/test_clarifier.py index 6cb5357113..cf7a8fc439 100644 --- a/watertap/unit_models/tests/test_clarifier.py +++ b/watertap/unit_models/tests/test_clarifier.py @@ -13,25 +13,33 @@ Tests for clarifier. """ __author__ = "Chenyu Wang" +import pytest from pyomo.environ import ( ConcreteModel, units, + Suffix, + TransformationFactory, ) from idaes.core import ( FlowsheetBlock, ) +from idaes.core.util.scaling import ( + get_jacobian, + jacobian_cond, +) from watertap.core.solvers import get_solver from watertap.unit_models.tests.unit_test_harness import UnitTestHarness import idaes.core.util.scaling as iscale -from watertap.unit_models.clarifier import Clarifier +from watertap.unit_models.clarifier import Clarifier, ClarifierScaler from idaes.models.unit_models.separator import SplittingType from watertap.property_models.unit_specific.activated_sludge.asm1_properties import ( ASM1ParameterBlock, + ASM1PropertiesScaler, ) from idaes.core import UnitModelCostingBlock @@ -283,3 +291,337 @@ def configure(self): } return m + + +class TestClarifierScaler: + @pytest.fixture + def model(self): + m = ConcreteModel() + m.fs = FlowsheetBlock(dynamic=False) + + m.fs.properties = ASM1ParameterBlock() + + m.fs.unit = Clarifier( + property_package=m.fs.properties, + outlet_list=["underflow", "effluent"], + split_basis=SplittingType.componentFlow, + ) + + m.fs.unit.inlet.temperature.fix(298.15 * units.K) + m.fs.unit.inlet.pressure.fix(1 * units.atm) + m.fs.unit.inlet.flow_vol.fix(18446 * units.m**3 / units.day) + + m.fs.unit.inlet.conc_mass_comp[0, "S_I"].fix(27 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_S"].fix(58 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_I"].fix(92 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_S"].fix(363 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BH"].fix(50 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BA"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_P"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_O"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NO"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NH"].fix(23 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_ND"].fix(5 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_ND"].fix(16 * units.g / units.m**3) + + # Alkalinity was given in mg/L based on C + m.fs.unit.inlet.alkalinity[0].fix(7 * units.mol / units.m**3) + + # Unit option + m.fs.unit.split_fraction[0, "effluent", "H2O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_I"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_S"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_I"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_S"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BH"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BA"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_P"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NO"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NH"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_ND"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_ND"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_ALK"].fix(0.993) + + return m + + @pytest.mark.component + def test_variable_scaling_routine(self, model): + scaler = model.fs.unit.default_scaler() + + assert isinstance(scaler, ClarifierScaler) + + scaler.variable_scaling_routine(model.fs.unit) + + # Inlet state + sfx_in = model.fs.unit.mixed_state[0].scaling_factor + assert isinstance(sfx_in, Suffix) + assert len(sfx_in) == 3 + assert sfx_in[model.fs.unit.mixed_state[0].flow_vol] == pytest.approx( + 1e1, rel=1e-8 + ) + assert sfx_in[model.fs.unit.mixed_state[0].pressure] == pytest.approx( + 1e-6, rel=1e-8 + ) + assert sfx_in[model.fs.unit.mixed_state[0].temperature] == pytest.approx( + 1e-1, rel=1e-8 + ) + + # Outlet state - should be the same as the inlet + sfx_underflow = model.fs.unit.underflow_state[0].scaling_factor + assert isinstance(sfx_underflow, Suffix) + assert len(sfx_underflow) == 3 + assert sfx_underflow[ + model.fs.unit.underflow_state[0].flow_vol + ] == pytest.approx(1e1, rel=1e-8) + assert sfx_underflow[ + model.fs.unit.underflow_state[0].pressure + ] == pytest.approx(1e-6, rel=1e-8) + assert sfx_underflow[ + model.fs.unit.underflow_state[0].temperature + ] == pytest.approx(1e-1, rel=1e-8) + + sfx_effluent = model.fs.unit.effluent_state[0].scaling_factor + assert isinstance(sfx_effluent, Suffix) + assert len(sfx_effluent) == 3 + assert sfx_effluent[model.fs.unit.effluent_state[0].flow_vol] == pytest.approx( + 1e1, rel=1e-8 + ) + assert sfx_effluent[model.fs.unit.effluent_state[0].pressure] == pytest.approx( + 1e-6, rel=1e-8 + ) + assert sfx_effluent[ + model.fs.unit.effluent_state[0].temperature + ] == pytest.approx(1e-1, rel=1e-8) + + # Check that unit model has scaling factors + sfx_unit = model.fs.unit.scaling_factor + assert isinstance(sfx_unit, Suffix) + assert len(sfx_unit) == 1 + assert sfx_unit[model.fs.unit.surface_area] == pytest.approx(1e-3, rel=1e-3) + + @pytest.mark.component + def test_constraint_scaling_routine(self, model): + scaler = model.fs.unit.default_scaler() + + assert isinstance(scaler, ClarifierScaler) + + scaler.constraint_scaling_routine(model.fs.unit) + + sfx_unit = model.fs.unit.scaling_factor + assert isinstance(sfx_unit, Suffix) + assert len(sfx_unit) == 47 + assert sfx_unit[model.fs.unit.rule_electricity_consumption[0]] == pytest.approx( + 0.162636886, rel=1e-8 + ) + + @pytest.mark.component + def test_scale_model(self, model): + scaler = model.fs.unit.default_scaler() + + assert isinstance(scaler, ClarifierScaler) + + scaler.scale_model(model.fs.unit) + + # Inlet state + sfx_in = model.fs.unit.mixed_state[0].scaling_factor + assert isinstance(sfx_in, Suffix) + assert len(sfx_in) == 3 + assert sfx_in[model.fs.unit.mixed_state[0].flow_vol] == pytest.approx( + 1e1, rel=1e-8 + ) + assert sfx_in[model.fs.unit.mixed_state[0].pressure] == pytest.approx( + 1e-6, rel=1e-8 + ) + assert sfx_in[model.fs.unit.mixed_state[0].temperature] == pytest.approx( + 1e-1, rel=1e-8 + ) + + # Outlet state - should be the same as the inlet + sfx_underflow = model.fs.unit.underflow_state[0].scaling_factor + assert isinstance(sfx_underflow, Suffix) + assert len(sfx_underflow) == 3 + assert sfx_underflow[ + model.fs.unit.underflow_state[0].flow_vol + ] == pytest.approx(1e1, rel=1e-8) + assert sfx_underflow[ + model.fs.unit.underflow_state[0].pressure + ] == pytest.approx(1e-6, rel=1e-8) + assert sfx_underflow[ + model.fs.unit.underflow_state[0].temperature + ] == pytest.approx(1e-1, rel=1e-8) + + sfx_effluent = model.fs.unit.underflow_state[0].scaling_factor + assert isinstance(sfx_effluent, Suffix) + assert len(sfx_effluent) == 3 + assert sfx_effluent[model.fs.unit.underflow_state[0].flow_vol] == pytest.approx( + 1e1, rel=1e-8 + ) + assert sfx_effluent[model.fs.unit.underflow_state[0].pressure] == pytest.approx( + 1e-6, rel=1e-8 + ) + assert sfx_effluent[ + model.fs.unit.underflow_state[0].temperature + ] == pytest.approx(1e-1, rel=1e-8) + + # Check that unit model has scaling factors + sfx_unit = model.fs.unit.scaling_factor + assert isinstance(sfx_unit, Suffix) + assert len(sfx_unit) == 48 + assert sfx_unit[model.fs.unit.surface_area] == pytest.approx(1e-3, rel=1e-3) + assert sfx_unit[model.fs.unit.rule_electricity_consumption[0]] == pytest.approx( + 0.162636886, rel=1e-8 + ) + + # TODO: Remove test once iscale is deprecated + @pytest.mark.integration + def test_example_case_iscale(self): + m = ConcreteModel() + m.fs = FlowsheetBlock(dynamic=False) + + m.fs.properties = ASM1ParameterBlock() + + m.fs.unit = Clarifier( + property_package=m.fs.properties, + outlet_list=["underflow", "effluent"], + split_basis=SplittingType.componentFlow, + ) + + m.fs.unit.inlet.temperature.fix(298.15 * units.K) + m.fs.unit.inlet.pressure.fix(1 * units.atm) + m.fs.unit.inlet.flow_vol.fix(18446 * units.m**3 / units.day) + + m.fs.unit.inlet.conc_mass_comp[0, "S_I"].fix(27 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_S"].fix(58 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_I"].fix(92 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_S"].fix(363 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BH"].fix(50 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BA"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_P"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_O"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NO"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NH"].fix(23 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_ND"].fix(5 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_ND"].fix(16 * units.g / units.m**3) + + # Alkalinity was given in mg/L based on C + m.fs.unit.inlet.alkalinity[0].fix(7 * units.mol / units.m**3) + + # Unit option + m.fs.unit.split_fraction[0, "effluent", "H2O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_I"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_S"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_I"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_S"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BH"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BA"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_P"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NO"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NH"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_ND"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_ND"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_ALK"].fix(0.993) + + # Set scaling factors for badly scaled variables + iscale.set_scaling_factor(m.fs.unit.underflow_state[0.0].pressure, 1e-5) + iscale.set_scaling_factor( + m.fs.unit.underflow_state[0.0].conc_mass_comp["X_BA"], 1e3 + ) + iscale.set_scaling_factor( + m.fs.unit.underflow_state[0.0].conc_mass_comp["X_P"], 1e3 + ) + iscale.set_scaling_factor( + m.fs.unit.underflow_state[0.0].conc_mass_comp["S_O"], 1e3 + ) + iscale.set_scaling_factor( + m.fs.unit.underflow_state[0.0].conc_mass_comp["S_NO"], 1e3 + ) + iscale.set_scaling_factor(m.fs.unit.effluent_state[0.0].pressure, 1e-5) + iscale.set_scaling_factor( + m.fs.unit.effluent_state[0.0].conc_mass_comp["X_BA"], 1e7 + ) + iscale.set_scaling_factor( + m.fs.unit.effluent_state[0.0].conc_mass_comp["X_P"], 1e7 + ) + iscale.set_scaling_factor( + m.fs.unit.effluent_state[0.0].conc_mass_comp["S_O"], 1e7 + ) + iscale.set_scaling_factor( + m.fs.unit.effluent_state[0.0].conc_mass_comp["S_NO"], 1e7 + ) + + iscale.calculate_scaling_factors(m.fs.unit) + + # Check condition number to confirm scaling + sm = TransformationFactory("core.scale_model").create_using(m, rename=False) + jac, _ = get_jacobian(sm, scaled=False) + assert (jacobian_cond(jac=jac, scaled=False)) == pytest.approx( + 2.955746851e9, rel=1e-3 + ) + + @pytest.mark.integration + def test_example_case_scaler(self): + m = ConcreteModel() + m.fs = FlowsheetBlock(dynamic=False) + + m.fs.properties = ASM1ParameterBlock() + + m.fs.unit = Clarifier( + property_package=m.fs.properties, + outlet_list=["underflow", "effluent"], + split_basis=SplittingType.componentFlow, + ) + + m.fs.unit.inlet.temperature.fix(298.15 * units.K) + m.fs.unit.inlet.pressure.fix(1 * units.atm) + m.fs.unit.inlet.flow_vol.fix(18446 * units.m**3 / units.day) + + m.fs.unit.inlet.conc_mass_comp[0, "S_I"].fix(27 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_S"].fix(58 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_I"].fix(92 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_S"].fix(363 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BH"].fix(50 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BA"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_P"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_O"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NO"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NH"].fix(23 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_ND"].fix(5 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_ND"].fix(16 * units.g / units.m**3) + + # Alkalinity was given in mg/L based on C + m.fs.unit.inlet.alkalinity[0].fix(7 * units.mol / units.m**3) + + # Unit option + m.fs.unit.split_fraction[0, "effluent", "H2O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_I"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_S"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_I"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_S"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BH"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BA"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_P"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NO"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NH"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_ND"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_ND"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_ALK"].fix(0.993) + + scaler = ClarifierScaler() + scaler.scale_model( + m.fs.unit, + submodel_scalers={ + m.fs.unit.mixed_state: ASM1PropertiesScaler, + m.fs.unit.underflow_state: ASM1PropertiesScaler, + m.fs.unit.effluent_state: ASM1PropertiesScaler, + }, + ) + + # Check condition number to confirm scaling + sm = TransformationFactory("core.scale_model").create_using(m, rename=False) + jac, _ = get_jacobian(sm, scaled=False) + assert (jacobian_cond(jac=jac, scaled=False)) == pytest.approx( + 2.0028333e4, rel=1e-3 + ) From 90f66f11520da31429dca11fd8e705903048058c Mon Sep 17 00:00:00 2001 From: MarcusHolly Date: Fri, 10 Jan 2025 11:13:52 -0500 Subject: [PATCH 2/2] Add testing for ScalingProfiler --- watertap/unit_models/tests/test_clarifier.py | 165 ++++++++++++++++++- 1 file changed, 164 insertions(+), 1 deletion(-) diff --git a/watertap/unit_models/tests/test_clarifier.py b/watertap/unit_models/tests/test_clarifier.py index cf7a8fc439..0c6aaaaed1 100644 --- a/watertap/unit_models/tests/test_clarifier.py +++ b/watertap/unit_models/tests/test_clarifier.py @@ -13,6 +13,7 @@ Tests for clarifier. """ __author__ = "Chenyu Wang" +from io import StringIO import pytest from pyomo.environ import ( ConcreteModel, @@ -28,7 +29,7 @@ get_jacobian, jacobian_cond, ) - +from idaes.core.scaling.scaler_profiling import ScalingProfiler from watertap.core.solvers import get_solver from watertap.unit_models.tests.unit_test_harness import UnitTestHarness @@ -625,3 +626,165 @@ def test_example_case_scaler(self): assert (jacobian_cond(jac=jac, scaled=False)) == pytest.approx( 2.0028333e4, rel=1e-3 ) + + +def build_model(): + m = ConcreteModel() + m.fs = FlowsheetBlock(dynamic=False) + + m.fs.properties = ASM1ParameterBlock() + + m.fs.unit = Clarifier( + property_package=m.fs.properties, + outlet_list=["underflow", "effluent"], + split_basis=SplittingType.componentFlow, + ) + + m.fs.unit.inlet.temperature.fix(298.15 * units.K) + m.fs.unit.inlet.pressure.fix(1 * units.atm) + m.fs.unit.inlet.flow_vol.fix(18446 * units.m**3 / units.day) + + m.fs.unit.inlet.conc_mass_comp[0, "S_I"].fix(27 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_S"].fix(58 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_I"].fix(92 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_S"].fix(363 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BH"].fix(50 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_BA"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_P"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_O"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NO"].fix(1e-3 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_NH"].fix(23 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "S_ND"].fix(5 * units.g / units.m**3) + m.fs.unit.inlet.conc_mass_comp[0, "X_ND"].fix(16 * units.g / units.m**3) + + # Alkalinity was given in mg/L based on C + m.fs.unit.inlet.alkalinity[0].fix(7 * units.mol / units.m**3) + + # Unit option + m.fs.unit.split_fraction[0, "effluent", "H2O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_I"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_S"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_I"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_S"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BH"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_BA"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "X_P"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_O"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NO"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_NH"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "S_ND"].fix(0.993) + m.fs.unit.split_fraction[0, "effluent", "X_ND"].fix(0.5192) + m.fs.unit.split_fraction[0, "effluent", "S_ALK"].fix(0.993) + + solver = get_solver() + solver.solve(m) + + return m + + +def scale_vars_with_scalers(m): + scaler = ClarifierScaler() + scaler.scale_model( + m.fs.unit, + submodel_scalers={ + m.fs.unit.mixed_state: ASM1PropertiesScaler, + m.fs.unit.underflow_state: ASM1PropertiesScaler, + m.fs.unit.effluent_state: ASM1PropertiesScaler, + }, + ) + + +def scale_vars_with_iscale(m): + # Set scaling factors for badly scaled variables + iscale.set_scaling_factor(m.fs.unit.underflow_state[0.0].pressure, 1e-5) + iscale.set_scaling_factor( + m.fs.unit.underflow_state[0.0].conc_mass_comp["X_BA"], 1e3 + ) + iscale.set_scaling_factor(m.fs.unit.underflow_state[0.0].conc_mass_comp["X_P"], 1e3) + iscale.set_scaling_factor(m.fs.unit.underflow_state[0.0].conc_mass_comp["S_O"], 1e3) + iscale.set_scaling_factor( + m.fs.unit.underflow_state[0.0].conc_mass_comp["S_NO"], 1e3 + ) + iscale.set_scaling_factor(m.fs.unit.effluent_state[0.0].pressure, 1e-5) + iscale.set_scaling_factor(m.fs.unit.effluent_state[0.0].conc_mass_comp["X_BA"], 1e7) + iscale.set_scaling_factor(m.fs.unit.effluent_state[0.0].conc_mass_comp["X_P"], 1e7) + iscale.set_scaling_factor(m.fs.unit.effluent_state[0.0].conc_mass_comp["S_O"], 1e7) + iscale.set_scaling_factor(m.fs.unit.effluent_state[0.0].conc_mass_comp["S_NO"], 1e7) + + iscale.calculate_scaling_factors(m.fs.unit) + + +def perturb_solution(m): + m.fs.unit.inlet.flow_vol.fix(18446 * 0.8 * units.m**3 / units.day) + m.fs.unit.split_fraction[0, "effluent", "H2O"].fix(0.993 * 0.35) + m.fs.unit.inlet.conc_mass_comp[0, "S_I"].fix(27 * 1.5 * units.g / units.m**3) + + +@pytest.mark.requires_idaes_solver +@pytest.mark.unit +def test_scaling_profiler_with_scalers(): + sp = ScalingProfiler( + build_model=build_model, + user_scaling=scale_vars_with_scalers, + perturb_state=perturb_solution, + ) + + stream = StringIO() + + sp.report_scaling_profiles(stream=stream) + + expected = """ +============================================================================ +Scaling Profile Report +---------------------------------------------------------------------------- +Scaling Method || User Scaling || Perfect Scaling +Unscaled || 6.241E+06 | Solved 3 || +Vars Only || 1.359E+10 | Solved 3 || 1.356E+14 | Solved 1 +Harmonic || 1.359E+10 | Solved 3 || 1.228E+02 | Solved 1 +Inverse Sum || 1.359E+10 | Solved 3 || 6.636E+03 | Solved 1 +Inverse Root Sum Squares || 1.359E+10 | Solved 3 || 6.633E+03 | Solved 1 +Inverse Maximum || 1.359E+10 | Solved 3 || 6.636E+03 | Solved 1 +Inverse Minimum || 1.359E+10 | Solved 3 || 9.327E+01 | Solved 1 +Nominal L1 Norm || 1.359E+10 | Solved 3 || 1.817E+03 | Solved 1 +Nominal L2 Norm || 1.359E+10 | Solved 3 || 3.354E+03 | Solved 1 +Actual L1 Norm || 1.359E+10 | Solved 3 || 9.450E+01 | Solved 1 +Actual L2 Norm || 1.359E+10 | Solved 3 || 8.480E+01 | Solved 1 +============================================================================ +""" + + assert stream.getvalue() == expected + + +@pytest.mark.requires_idaes_solver +@pytest.mark.unit +def test_scaling_profiler_with_iscale(): + sp = ScalingProfiler( + build_model=build_model, + user_scaling=scale_vars_with_iscale, + perturb_state=perturb_solution, + ) + + stream = StringIO() + + sp.report_scaling_profiles(stream=stream) + + expected = """ +============================================================================ +Scaling Profile Report +---------------------------------------------------------------------------- +Scaling Method || User Scaling || Perfect Scaling +Unscaled || 6.241E+06 | Solved 3 || +Vars Only || 2.999E+09 | Solved 2 || 1.356E+14 | Solved 1 +Harmonic || 1.027E+06 | Solved 2 || 1.228E+02 | Solved 1 +Inverse Sum || 8.148E+06 | Solved 2 || 6.636E+03 | Solved 1 +Inverse Root Sum Squares || 8.114E+06 | Solved 2 || 6.633E+03 | Solved 1 +Inverse Maximum || 8.139E+06 | Solved 2 || 6.636E+03 | Solved 1 +Inverse Minimum || 1.257E+06 | Solved 2 || 9.327E+01 | Solved 1 +Nominal L1 Norm || 1.632E+09 | Solved 2 || 1.817E+03 | Solved 1 +Nominal L2 Norm || 1.972E+09 | Solved 2 || 3.354E+03 | Solved 1 +Actual L1 Norm || 1.776E+05 | Solved 2 || 9.450E+01 | Solved 1 +Actual L2 Norm || 1.723E+05 | Solved 2 || 8.480E+01 | Solved 1 +============================================================================ +""" + + assert stream.getvalue() == expected